aboutsummaryrefslogtreecommitdiff
path: root/engines
diff options
context:
space:
mode:
Diffstat (limited to 'engines')
-rw-r--r--engines/sci/engine/kpathing.cpp134
1 files changed, 49 insertions, 85 deletions
diff --git a/engines/sci/engine/kpathing.cpp b/engines/sci/engine/kpathing.cpp
index 315a9ecc79..51cb611b1b 100644
--- a/engines/sci/engine/kpathing.cpp
+++ b/engines/sci/engine/kpathing.cpp
@@ -23,10 +23,6 @@
*
*/
-/* Detailed information on the implementation can be found in the report
-** which can be downloaded from FIXME.
-*/
-
#include "sci/engine/state.h"
#include "sci/engine/aatree.h"
#include "sci/engine/kernel.h"
@@ -70,7 +66,7 @@ enum {
CONT_INSIDE = 2
};
-#define HUGE_DISTANCE 1e10
+#define HUGE_DISTANCE 0xFFFFFFFF
#define VERTEX_HAS_EDGES(V) ((V) != CLIST_NEXT(V))
@@ -86,26 +82,23 @@ struct FloatPoint {
FloatPoint() : x(0), y(0) {}
FloatPoint(float x_, float y_) : x(x_), y(y_) {}
+ Common::Point toPoint() {
+ return Common::Point((int16)roundf(x), (int16)roundf(y));
+ }
+
float x, y;
};
-FloatPoint toFloatPoint(const Common::Point &p) {
- return FloatPoint(p.x, p.y);
-}
-
struct Vertex {
// Location
Common::Point v;
- // Index
- int idx;
-
// Vertex circular list entry
Vertex *_next; // next element
Vertex *_prev; // previous element
// Distance from starting vertex
- float dist;
+ uint32 dist;
// Previous vertex in shortest path
Vertex *path_prev;
@@ -229,7 +222,7 @@ struct PathfindingState {
// Start and end points for pathfinding
Vertex *vertex_start, *vertex_end;
- // Array to quickly find vertices by idx
+ // Array of all vertices, used for sorting
Vertex **vertex_index;
// Total number of vertices
@@ -784,26 +777,23 @@ static bool visible(Vertex *vertex, Vertex *vertex_prev, bool visible, const Edg
return true;
}
+/**
+ * Returns a list of all vertices that are visible from a particular vertex.
+ * @param s the pathfinding state
+ * @param vert the vertex
+ * @return list of vertices that are visible from vert
+ */
static VertexList *visible_vertices(PathfindingState *s, Vertex *vert) {
- // Determines all vertices that are visible from a particular vertex and
- // updates the visibility matrix
- // Parameters: (PathfindingState *) s: The pathfinding state
- // (Vertex *) vert: The vertex
EdgeAATree tree;
VertexList *visVerts = new VertexList();
const Common::Point &p = vert->v;
- Polygon *polygon;
- int i;
- int is_visible;
- Vertex **vert_sorted = (Vertex**)sci_malloc(sizeof(Vertex *) * s->vertices);
// Sort vertices by angle (first) and distance (second)
- memcpy(vert_sorted, s->vertex_index, sizeof(Vertex *) * s->vertices);
vertex_cur = vert;
- qsort(vert_sorted, s->vertices, sizeof(Vertex *), vertex_compare);
+ qsort(s->vertex_index, s->vertices, sizeof(Vertex *), vertex_compare);
for (PolygonList::iterator it = s->polygons.begin(); it != s->polygons.end(); ++it) {
- polygon = *it;
+ Polygon *polygon = *it;
Vertex *vertex;
vertex = polygon->vertices.first();
@@ -821,72 +811,50 @@ static VertexList *visible_vertices(PathfindingState *s, Vertex *vert) {
}
}
- is_visible = 1;
+ int is_visible = 1;
// The first vertex will be vertex_cur, so we skip it
- for (i = 1; i < s->vertices; i++) {
+ for (int i = 1; i < s->vertices; i++) {
Vertex *v1;
// Compute visibility of vertex_index[i]
- is_visible = visible(vert_sorted[i], vert_sorted[i - 1], is_visible, tree);
+ is_visible = visible(s->vertex_index[i], s->vertex_index[i - 1], is_visible, tree);
// Update visibility matrix
if (is_visible)
- visVerts->push_front(vert_sorted[i]);
+ visVerts->push_front(s->vertex_index[i]);
// Delete anti-clockwise edges from tree
- v1 = CLIST_PREV(vert_sorted[i]);
- if (left(p, vert_sorted[i]->v, v1->v)) {
+ v1 = CLIST_PREV(s->vertex_index[i]);
+ if (left(p, s->vertex_index[i]->v, v1->v)) {
if (!tree.remove(v1))
sciprintf("[avoidpath] Error: failed to remove edge from tree\n");
}
- v1 = CLIST_NEXT(vert_sorted[i]);
- if (left(p, vert_sorted[i]->v, v1->v)) {
- if (!tree.remove(vert_sorted[i]))
+ v1 = CLIST_NEXT(s->vertex_index[i]);
+ if (left(p, s->vertex_index[i]->v, v1->v)) {
+ if (!tree.remove(s->vertex_index[i]))
sciprintf("[avoidpath] Error: failed to remove edge from tree\n");
}
// Add clockwise edges of collinear vertices when sweeping line moves
- if ((i < s->vertices - 1) && !collinear(p, vert_sorted[i]->v, vert_sorted[i + 1]->v)) {
+ if ((i < s->vertices - 1) && !collinear(p, s->vertex_index[i]->v, s->vertex_index[i + 1]->v)) {
int j;
- for (j = i; (j >= 1) && collinear(p, vert_sorted[i]->v, vert_sorted[j]->v); j--) {
- v1 = CLIST_PREV(vert_sorted[j]);
- if (left(vert_sorted[j]->v, p, v1->v))
+ for (j = i; (j >= 1) && collinear(p, s->vertex_index[i]->v, s->vertex_index[j]->v); j--) {
+ v1 = CLIST_PREV(s->vertex_index[j]);
+ if (left(s->vertex_index[j]->v, p, v1->v))
tree.insert(v1);
- v1 = CLIST_NEXT(vert_sorted[j]);
- if (left(vert_sorted[j]->v, p, v1->v))
- tree.insert(vert_sorted[j]);
+ v1 = CLIST_NEXT(s->vertex_index[j]);
+ if (left(s->vertex_index[j]->v, p, v1->v))
+ tree.insert(s->vertex_index[j]);
}
}
}
- free(vert_sorted);
-
return visVerts;
}
-/**
- * Computes the distance between two FloatPoints.
- */
-static float distance(const FloatPoint &a, const FloatPoint &b) {
- float w = a.x - b.x;
- float h = a.y - b.y;
-
- return sqrt(w * w + h * h);
-}
-
-/**
- * Computes the square of the distance between two FloatPoints.
- */
-static float distanceSqr(const FloatPoint &a, const FloatPoint &b) {
- float w = a.x - b.x;
- float h = a.y - b.y;
-
- return w * w + h * h;
-}
-
static bool point_on_screen_border(const Common::Point &p) {
// Determines if a point lies on the screen border
// Parameters: (const Common::Point &) p: The point
@@ -946,24 +914,21 @@ static int near_point(const Common::Point &p, Polygon *polygon, Common::Point *r
// (Common::Point) *ret: The near point of p in polygon on success
Vertex *vertex;
FloatPoint near_p;
- float dist = HUGE_DISTANCE;
+ uint32 dist = HUGE_DISTANCE;
CLIST_FOREACH(vertex, &polygon->vertices) {
const Common::Point &p1 = vertex->v;
const Common::Point &p2 = CLIST_NEXT(vertex)->v;
- float w, h, l, u;
+ float u;
FloatPoint new_point;
- float new_dist;
+ uint32 new_dist;
// Ignore edges on the screen border
if (edge_on_screen_border(p1, p2))
continue;
// Compute near point
- w = p2.x - p1.x;
- h = p2.y - p1.y;
- l = sqrt(w * w + h * h);
- u = ((p.x - p1.x) * (p2.x - p1.x) + (p.y - p1.y) * (p2.y - p1.y)) / (l * l);
+ u = ((p.x - p1.x) * (p2.x - p1.x) + (p.y - p1.y) * (p2.y - p1.y)) / (float)p1.sqrDist(p2);
// Clip to edge
if (u < 0.0f)
@@ -974,7 +939,7 @@ static int near_point(const Common::Point &p, Polygon *polygon, Common::Point *r
new_point.x = p1.x + u * (p2.x - p1.x);
new_point.y = p1.y + u * (p2.y - p1.y);
- new_dist = distanceSqr(toFloatPoint(p), new_point);
+ new_dist = p.sqrDist(new_point.toPoint());
if (new_dist < dist) {
near_p = new_point;
@@ -1038,14 +1003,14 @@ static int nearest_intersection(PathfindingState *s, const Common::Point &p, con
Polygon *polygon = 0;
FloatPoint isec;
Polygon *ipolygon = 0;
- float dist = HUGE_DISTANCE;
+ uint32 dist = HUGE_DISTANCE;
for (PolygonList::iterator it = s->polygons.begin(); it != s->polygons.end(); ++it) {
polygon = *it;
Vertex *vertex;
CLIST_FOREACH(vertex, &polygon->vertices) {
- float new_dist;
+ uint32 new_dist;
FloatPoint new_isec;
// Check for intersection with vertex
@@ -1069,7 +1034,7 @@ static int nearest_intersection(PathfindingState *s, const Common::Point &p, con
continue;
}
- new_dist = distanceSqr(toFloatPoint(p), new_isec);
+ new_dist = p.sqrDist(new_isec.toPoint());
if (new_dist < dist) {
ipolygon = polygon;
isec = new_isec;
@@ -1388,7 +1353,6 @@ static PathfindingState *convert_polygon_set(EngineState *s, reg_t poly_list, Co
Vertex *vertex;
CLIST_FOREACH(vertex, &polygon->vertices) {
- vertex->idx = count;
pf_s->vertex_index[count++] = vertex;
}
}
@@ -1450,14 +1414,14 @@ static void dijkstra(PathfindingState *s) {
}
}
- s->vertex_start->dist = 0.0f;
+ s->vertex_start->dist = 0;
// Loop until we find vertex_end
while (1) {
// Find vertex at shortest distance from set done
VertexList::iterator vertex_min_it = remain.end();
Vertex *vertex_min = 0;
- float min = HUGE_DISTANCE;
+ uint32 min = HUGE_DISTANCE;
for (VertexList::iterator it = remain.begin(); it != remain.end(); ++it) {
Vertex *vertex = *it;
if (vertex->dist < min) {
@@ -1483,14 +1447,14 @@ static void dijkstra(PathfindingState *s) {
VertexList *visVerts = visible_vertices(s, vertex_min);
for (VertexList::iterator it = visVerts->begin(); it != visVerts->end(); ++it) {
- float new_dist;
+ uint32 new_dist;
Vertex *vertex = *it;
// Avoid plotting path along screen edge
if ((vertex != s->vertex_end) && point_on_screen_border(vertex->v))
continue;
- new_dist = vertex_min->dist + distance(toFloatPoint(vertex_min->v), toFloatPoint(vertex->v));
+ new_dist = vertex_min->dist + (uint32)sqrt((float)vertex_min->v.sqrDist(vertex->v));
if (new_dist < vertex->dist) {
vertex->dist = new_dist;
vertex->path_prev = vertex_min;
@@ -1620,12 +1584,6 @@ reg_t kAvoidPath(EngineState *s, int funct_nr, int argc, reg_t *argv) {
p = convert_polygon_set(s, poly_list, start, end, opt);
- if (intersecting_polygons(p)) {
- sciprintf("[avoidpath] Error: input set contains (self-)intersecting polygons\n");
- delete p;
- p = NULL;
- }
-
if (!p) {
byte *oref;
sciprintf("[avoidpath] Error: pathfinding failed for following input:\n");
@@ -1641,6 +1599,12 @@ reg_t kAvoidPath(EngineState *s, int funct_nr, int argc, reg_t *argv) {
return output;
}
+ if (intersecting_polygons(p)) {
+ sciprintf("[avoidpath] Error: input set contains (self-)intersecting polygons\n");
+ delete p;
+ p = NULL;
+ }
+
dijkstra(p);
output = output_path(p, s);