diff options
Diffstat (limited to 'engines/sci/engine')
-rw-r--r-- | engines/sci/engine/kpathing.cpp | 92 |
1 files changed, 77 insertions, 15 deletions
diff --git a/engines/sci/engine/kpathing.cpp b/engines/sci/engine/kpathing.cpp index 5c0c4bcfa3..b5f4a90b13 100644 --- a/engines/sci/engine/kpathing.cpp +++ b/engines/sci/engine/kpathing.cpp @@ -126,6 +126,17 @@ public: typedef Common::List<Vertex *> VertexList; +/* Circular list definitions. */ + +#define CLIST_FOREACH(var, head) \ + for ((var) = (head)->first(); \ + (var); \ + (var) = ((var)->_next == (head)->first() ? \ + NULL : (var)->_next)) + +/* Circular list access methods. */ +#define CLIST_NEXT(elm) ((elm)->_next) +#define CLIST_PREV(elm) ((elm)->_prev) class CircularVertexList { public: @@ -134,7 +145,7 @@ public: public: CircularVertexList() : _head(0) {} - Vertex *first() { + Vertex *first() const { return _head; } @@ -172,6 +183,14 @@ public: return _head == NULL; } + uint size() const { + int n = 0; + Vertex *v; + CLIST_FOREACH(v, this) + ++n; + return n; + } + /** * Reverse the order of the elements in this circular list. */ @@ -187,19 +206,6 @@ public: } }; -/* Circular list definitions. */ - -#define CLIST_FOREACH(var, head) \ - for ((var) = (head)->first(); \ - (var); \ - (var) = ((var)->_next == (head)->first() ? \ - NULL : (var)->_next)) - -/* Circular list access methods. */ -#define CLIST_NEXT(elm) ((elm)->_next) -#define CLIST_PREV(elm) ((elm)->_prev) - - struct Polygon { // SCI polygon type int type; @@ -1217,6 +1223,56 @@ static Polygon *convert_polygon(EngineState *s, reg_t polygon) { return poly; } +// WORKAROUND: intersecting polygons in Longbow, room 210. +static void fixLongbowRoom210(PathfindingState *s, const Common::Point &start, const Common::Point &end) { + Polygon *barred = NULL; + Polygon *total = NULL; + + // Find the intersecting polygons + for (PolygonList::iterator it = s->polygons.begin(); it != s->polygons.end(); ++it) { + Polygon *polygon = *it; + assert(polygon); + + if ((polygon->type == POLY_BARRED_ACCESS) && (polygon->vertices.size() == 11) + && (polygon->vertices.first()->v == Common::Point(319, 161))) + barred = polygon; + else if ((polygon->type == POLY_TOTAL_ACCESS) && (polygon->vertices.size() == 8) + && (polygon->vertices.first()->v == Common::Point(313, 58))) + total = polygon; + } + + if (!barred || !total) + return; + + debug(1, "[avoidpath] Applying fix for intersecting polygons in Longbow, room 210"); + + // If the start or end point is contained in the total access polygon, removing that + // polygon is sufficient. Otherwise we merge the total and barred access polygons. + bool both_outside = (contained(start, total) == CONT_OUTSIDE) && (contained(end, total) == CONT_OUTSIDE); + + s->polygons.remove(total); + delete total; + + if (both_outside) { + int points[28] = { + 224, 159, 223, 162 ,194, 173 ,107, 173, 74, 162, 67, 156, 2, 58, + 63, 160, 0, 160, 0, 0, 319, 0, 319, 161, 228, 161, 313, 58 + }; + + s->polygons.remove(barred); + delete barred; + + barred = new Polygon(POLY_BARRED_ACCESS); + + for (int i = 0; i < 14; i++) { + Vertex *vertex = new Vertex(Common::Point(points[i * 2], points[i * 2 + 1])); + barred->vertices.insertHead(vertex); + } + + s->polygons.push_front(barred); + } +} + static void change_polygons_opt_0(PathfindingState *s) { // Changes the polygon list for optimization level 0 (used for keyboard // support). Totally accessible polygons are removed and near-point @@ -1275,7 +1331,7 @@ static PathfindingState *convert_polygon_set(EngineState *s, reg_t poly_list, Co if (dup == node) { // Polygon is not a duplicate, so convert it polygon = convert_polygon(s, node->value); - pf_s->polygons.push_front(polygon); + pf_s->polygons.push_back(polygon); count += KP_UINT(GET_SEL32(node->value, size)); } @@ -1321,6 +1377,12 @@ static PathfindingState *convert_polygon_set(EngineState *s, reg_t poly_list, Co pf_s->_appendPoint = new Common::Point(end); } + if (s->_gameName == "Longbow") { + // FIXME: implement function to get current room number + if ((KP_UINT(s->script_000->locals_block->locals[13]) == 210)) + fixLongbowRoom210(pf_s, new_start, new_end); + } + // Merge start and end points into polygon set pf_s->vertex_start = merge_point(pf_s, new_start); pf_s->vertex_end = merge_point(pf_s, new_end); |