aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--problem.cpp69
-rw-r--r--problem.hpp7
2 files changed, 51 insertions, 25 deletions
diff --git a/problem.cpp b/problem.cpp
index 664afb3..de6461e 100644
--- a/problem.cpp
+++ b/problem.cpp
@@ -81,7 +81,8 @@ bool hilare_a_mvt::intersects(const problem &p) const {
// IMPLEMENTATION FOR CLASS SOLUTION //
// ================================= //
-solution solution::direct_sol(const hilare_a &pos_a, const hilare_a &pos_b) {
+vector<solution> solution::direct_sol(const hilare_a &pos_a, const hilare_a &pos_b) {
+ vector<solution> ret;
// première famille de mouvements :
// - trouver les quatre droites tangentes aux deux cercles canoniques
@@ -128,34 +129,43 @@ solution solution::direct_sol(const hilare_a &pos_a, const hilare_a &pos_b) {
vec w = l.proj(ccb);
double domega1 = (v - cca).angle() - (pos_a.pos_trolley() - cca).angle();
+ if (domega1 > M_PI) domega1 -= 2 * M_PI;
+ if (domega1 < -M_PI) domega1 += 2 * M_PI;
double dtheta1 = pos_a.phi;
double dtheta2 = -pos_b.phi;
double domega2 = (pos_b.pos_trolley() - ccb).angle() - (w - ccb).angle();
+ if (domega2 > M_PI) domega2 -= 2 * M_PI;
+ if (domega2 < -M_PI) domega2 += 2 * M_PI;
+
double xx = pos_a.theta + domega1 + dtheta1 + dtheta2 + domega2 - pos_b.theta;
cout << "domega1: " << domega1
<< ", domega2: " << domega2
<< ", xx:" << xx << endl;
+
if (fabs(xx) < 0.01 || fabs(xx - 2*M_PI) < 0.01 && fabs(xx + 2*M_PI) < 0.01) {
vector<hilare_a_mvt> sol;
+ vec p1 = cca + vec::from_polar((pos_a.pos() - cca).norm(), (pos_a.pos() - cca).angle() + domega1);
+ vec p2 = ccb + vec::from_polar((pos_b.pos() - ccb).norm(), (pos_b.pos() - ccb).angle() - domega2);
+
hilare_a_mvt r1;
+ r1.dtheta_before = 0;
r1.is_arc = true;
+ r1.center = cca;
+ r1.domega = domega1;
r1.from = pos_a;
r1.to = pos_a;
- r1.to.x = v.x; r1.to.y = v.y;
+ r1.to.x = p1.x; r1.to.y = p1.y;
r1.to.theta = r1.from.theta + domega1;
- r1.center = cca;
- r1.domega = domega1;
- r1.dtheta_before = 0;
sol.push_back(r1);
hilare_a_mvt t;
- t.from = r1.to;
- t.to = t.from;
- t.to.x = w.x; t.to.y = w.y; t.to.phi = 0;
t.is_arc = false;
t.ds = (w - v).norm();
- t.dtheta_before = t.from.phi;
+ t.dtheta_before = r1.to.phi;
+ t.from = r1.to;
+ t.to = t.from; t.to.theta = t.from.theta + t.dtheta_before;
+ t.to.x = p2.x; t.to.y = p2.y; t.to.phi = 0;
sol.push_back(t);
hilare_a_mvt r2;
@@ -167,11 +177,11 @@ solution solution::direct_sol(const hilare_a &pos_a, const hilare_a &pos_b) {
r2.domega = domega2;
sol.push_back(r2);
- return solution(sol);
+ ret.push_back(sol);
}
}
- return solution(); // empty solution
+ return ret;
}
bool solution::intersects(const problem &p) const {
@@ -181,6 +191,14 @@ bool solution::intersects(const problem &p) const {
return false;
}
+double solution::length() {
+ double x;
+ for (auto& m: movement) {
+ x += m.length();
+ }
+ return x;
+}
+
// =============================== //
// IMPLEMENTATION FOR CLASS SOLVER //
// =============================== //
@@ -266,10 +284,7 @@ void solver_internal::initialize(const problem &p) {
pts.push_back(p.begin_pos);
pts.push_back(p.end_pos);
- solution ts = solution::direct_sol(p.begin_pos, p.end_pos);
- if (ts.movement.size() > 0 && !ts.intersects(p)) {
- paths[0][1] = ts;
- }
+ find_direct_path(0, 1, p);
}
solution solver_internal::try_find_solution() {
@@ -339,18 +354,24 @@ void solver_internal::step(const problem &p) {
rp.theta = frand(-M_PI, M_PI);
rp.phi = frand(-M_PI, M_PI);
+ pts.push_back(rp);
+
// try to connect to all existing points
- for (unsigned i = 0; i < pts.size(); i++) {
- solution s = solution::direct_sol(pts[i], rp);
- if (s.movement.size() > 0 && !s.intersects(p)) {
- paths[i][pts.size()] = s;
- }
- solution ss = solution::direct_sol(rp, pts[i]);
- if (ss.movement.size() > 0 && !ss.intersects(p)) {
- paths[pts.size()][i] = ss;
+ for (unsigned i = 0; i < pts.size() - 1; i++) {
+ find_direct_path(i, pts.size() - 1, p);
+ find_direct_path(pts.size() - 1, i, p);
+ }
+}
+
+void solver_internal::find_direct_path(int a, int b, const problem &p) {
+ vector<solution> s = solution::direct_sol(pts[a], pts[b]);
+ int best = -1;
+ for (int k = 0; k < s.size(); k++) {
+ if (s[k].movement.size() > 0 && !s[k].intersects(p)) {
+ if (best == -1 || s[k].length() < s[best].length()) best = k;
}
}
- pts.push_back(rp);
+ if (best != -1) paths[a][b] = s[best];
}
/* vim: set ts=4 sw=4 tw=0 noet :*/
diff --git a/problem.hpp b/problem.hpp
index 14ba475..4933ed6 100644
--- a/problem.hpp
+++ b/problem.hpp
@@ -91,10 +91,12 @@ struct solution {
solution(const std::vector<hilare_a_mvt> &m) : movement(m) {}
// simple direct solution from A to B, takes into account no obstacles
- static solution direct_sol(const hilare_a &pos_a, const hilare_a &pos_b);
+ static std::vector<solution> direct_sol(const hilare_a &pos_a, const hilare_a &pos_b);
// check if a solution intersects an obstacle of problem
bool intersects(const problem &p) const;
+
+ double length();
};
struct solver_internal {
@@ -106,6 +108,9 @@ struct solver_internal {
void initialize(const problem &p);
solution try_find_solution();
void step(const problem &p);
+
+ // internal
+ void find_direct_path(int a, int b, const problem &p);
};
class solver {