#ifndef TRAVEL_H
#define TRAVEL_H
#include "externs.h"
#include "exclude.h"
#include "los.h"
#include <stdio.h>
#include <string>
#include <vector>
#include <map>
class reader;
class writer;
enum run_check_type
{
RCHECK_LEFT,
RCHECK_FRONT,
RCHECK_RIGHT
};
enum run_dir_type
{
RDIR_UP = 0,
RDIR_UP_RIGHT,
RDIR_RIGHT,
RDIR_DOWN_RIGHT,
RDIR_DOWN,
RDIR_DOWN_LEFT,
RDIR_LEFT,
RDIR_UP_LEFT,
RDIR_REST
};
enum run_mode_type
{
RMODE_INTERLEVEL = -4, RMODE_EXPLORE_GREEDY = -3, RMODE_EXPLORE = -2, RMODE_TRAVEL = -1, RMODE_NOT_RUNNING = 0, RMODE_CONTINUE,
RMODE_START,
RMODE_REST_DURATION = 100
};
void initialise_travel();
void init_travel_terrain_check(bool check_race_equip = true);
void stop_running(void);
void travel_init_new_level();
unsigned char is_waypoint(const coord_def &p);
command_type direction_to_command(char x, char y);
bool is_resting(void);
#ifdef CLUA_BINDINGS
const char *trap_name(const coord_def &p);
#endif
void explore_pickup_event(int did_pickup, int tried_pickup);
bool feat_is_traversable(dungeon_feature_type feat);
bool is_unknown_stair(const coord_def &p,
dungeon_feature_type remembered_feat = NUM_REAL_FEATURES);
void find_travel_pos(const coord_def& youpos, char *move_x, char *move_y,
std::vector<coord_def>* coords = NULL);
bool is_travelsafe_square(const coord_def& c, bool ignore_hostile = false,
bool ignore_terrain_knowledge = false);
void start_explore(bool grab_items = false);
struct level_pos;
class level_id;
struct travel_target;
void start_translevel_travel(const travel_target &pos);
void start_translevel_travel(bool prompt_for_destination = true);
void start_travel(const coord_def& p);
command_type travel();
int travel_direction(unsigned char branch, int subdungeondepth);
void prevent_travel_to(const std::string &dungeon_feature_name);
void arrange_features(std::vector<coord_def> &features);
int level_distance(level_id first, level_id second);
bool can_travel_to(const level_id &lid);
bool can_travel_interlevel();
bool prompt_stop_explore(int es_why);
enum translevel_prompt_flags
{
TPF_NO_FLAGS = 0,
TPF_ALLOW_WAYPOINTS = 0x1,
TPF_ALLOW_UPDOWN = 0x2,
TPF_REMEMBER_TARGET = 0x4,
TPF_SHOW_ALL_BRANCHES = 0x8,
TPF_DEFAULT_OPTIONS = TPF_ALLOW_WAYPOINTS | TPF_ALLOW_UPDOWN
| TPF_REMEMBER_TARGET
};
travel_target prompt_translevel_target(int prompt_flags = TPF_DEFAULT_OPTIONS);
const int PD_TRAP = -42;
const int PD_EXCLUDED = -20099;
const int PD_EXCLUDED_RADIUS = -20100;
typedef int travel_distance_col[GYM];
typedef travel_distance_col travel_distance_grid_t[GXM];
extern travel_distance_grid_t travel_point_distance;
enum explore_stop_type
{
ES_NONE = 0x000,
ES_ITEM = 0x001,
ES_PICKUP = 0x002,
ES_GREEDY_PICKUP = 0x004,
ES_GREEDY_ITEM = 0x008,
ES_STAIR = 0x010,
ES_SHOP = 0x020,
ES_ALTAR = 0x040,
ES_PORTAL = 0x080,
ES_GLOWING_ITEM = 0x100,
ES_ARTEFACT = 0x200,
ES_RUNE = 0x400
};
struct level_pos
{
level_id id;
coord_def pos;
level_pos() : id(), pos()
{
pos.x = pos.y = -1;
}
level_pos(const level_id &lid, const coord_def &coord)
: id(lid), pos(coord)
{
}
level_pos(const level_id &lid)
: id(lid), pos()
{
pos.x = pos.y = -1;
}
static level_pos current();
bool operator == ( const level_pos &lp ) const
{
return id == lp.id && pos == lp.pos;
}
bool operator != ( const level_pos &lp ) const
{
return id != lp.id || pos != lp.pos;
}
bool operator < ( const level_pos &lp ) const
{
return (id < lp.id) || (id == lp.id && pos < lp.pos);
}
bool is_valid() const
{
return id.depth > -1 && pos.x != -1 && pos.y != -1;
}
bool is_on( const level_id _id)
{
return id == _id;
}
void clear()
{
id.clear();
pos = coord_def(-1, -1);
}
void save(writer&) const;
void load(reader&);
};
struct travel_target
{
level_pos p;
bool entrance_only;
travel_target(const level_pos &_pos, bool entry = false)
: p(_pos), entrance_only(entry)
{
}
travel_target()
: p(), entrance_only(false)
{
}
void clear()
{
p.clear();
entrance_only = false;
}
bool operator == (const travel_target &other) const
{
return p == other.p && entrance_only == other.entrance_only;
}
bool operator != (const travel_target &other) const
{
return !operator == (other);
}
};
class LevelStashes;
class explore_discoveries
{
public:
explore_discoveries();
void found_feature(const coord_def &pos, dungeon_feature_type grid);
void found_item(const coord_def &pos, const item_def &item);
bool prompt_stop() const;
private:
template <class Z> struct named_thing {
std::string name;
Z thing;
named_thing(const std::string &n, Z t) : name(n), thing(t) { }
operator std::string () const { return name; }
bool operator == (const named_thing<Z> &other) const
{
return (name == other.name);
}
};
int es_flags;
const LevelStashes *current_level;
std::vector< named_thing<item_def> > items;
std::vector< named_thing<int> > stairs;
std::vector< named_thing<int> > portals;
std::vector< named_thing<int> > shops;
std::vector< named_thing<int> > altars;
std::vector<std::string> marker_msgs;
std::vector<std::string> marked_feats;
private:
template <class C> void say_any(const C &coll, const char *stub) const;
template <class citer> bool has_duplicates(citer, citer) const;
std::string cleaned_feature_description(const coord_def &) const;
void add_item(const item_def &item);
void add_stair(const named_thing<int> &stair);
std::vector<std::string> apply_quantities(
const std::vector< named_thing<int> > &v) const;
bool merge_feature(
std::vector< named_thing<int> > &v,
const named_thing<int> &feat) const;
};
struct stair_info
{
public:
enum stair_type
{
PHYSICAL,
PLACEHOLDER
};
public:
coord_def position; dungeon_feature_type grid; level_pos destination; int distance; bool guessed_pos; stair_type type;
stair_info()
: position(-1, -1), grid(DNGN_FLOOR), destination(),
distance(-1), guessed_pos(true), type(PHYSICAL)
{
}
void clear_distance() { distance = -1; }
void save(writer&) const;
void load(reader&);
std::string describe() const;
bool can_travel() const { return (type == PHYSICAL); }
};
struct LevelInfo
{
LevelInfo() : stairs(), excludes(), stair_distances(), id()
{
}
void save(writer&) const;
void load(reader&, char minorVersion);
std::vector<stair_info> &get_stairs()
{
return stairs;
}
stair_info *get_stair(int x, int y);
stair_info *get_stair(const coord_def &pos);
bool empty() const;
bool know_stair(const coord_def &pos) const;
int get_stair_index(const coord_def &pos) const;
void clear_distances();
void set_level_excludes();
const std::vector<travel_exclude> &get_excludes() const
{
return excludes;
}
int distance_between(const stair_info *s1, const stair_info *s2) const;
void update();
void update_stair(const coord_def& stairpos, const level_pos &p,
bool guess = false);
void clear_stairs(dungeon_feature_type grid);
bool is_known_branch(unsigned char branch) const;
private:
static void get_stairs(std::vector<coord_def> &stairs);
void correct_stair_list(const std::vector<coord_def> &s);
void update_stair_distances();
void sync_all_branch_stairs();
void sync_branch_stairs(const stair_info *si);
void fixup();
private:
std::vector<stair_info> stairs;
std::vector<travel_exclude> excludes;
std::vector<short> stair_distances; level_id id;
friend class TravelCache;
private:
void create_placeholder_stair(const coord_def &, const level_pos &);
void resize_stair_distances();
};
const int TRAVEL_WAYPOINT_COUNT = 10;
class TravelCache
{
public:
void clear_distances();
LevelInfo& get_level_info(const level_id &lev)
{
LevelInfo &li = levels[lev];
li.id = lev;
return li;
}
LevelInfo *find_level_info(const level_id &lev)
{
std::map<level_id, LevelInfo>::iterator i = levels.find(lev);
return (i != levels.end()? &i->second : NULL);
}
bool know_stair(const coord_def &c) const;
bool know_level(const level_id &lev) const
{
return levels.find(lev) != levels.end();
}
const level_pos &get_waypoint(int number) const
{
return waypoints[number];
}
int get_waypoint_count() const;
void set_level_excludes();
void add_waypoint(int x = -1, int y = -1);
void delete_waypoint();
unsigned char is_waypoint(const level_pos &lp) const;
void list_waypoints() const;
void travel_to_waypoint(int number);
void update_waypoints() const;
void update();
void save(writer&) const;
void load(reader&, char minorVersion);
bool is_known_branch(unsigned char branch) const;
private:
void fixup_levels();
private:
typedef std::map<level_id, LevelInfo> travel_levels_map;
travel_levels_map levels;
level_pos waypoints[TRAVEL_WAYPOINT_COUNT];
};
class travel_pathfind
{
public:
travel_pathfind();
virtual ~travel_pathfind();
coord_def pathfind(run_mode_type rt);
void set_floodseed(const coord_def &seed, bool double_flood = false);
void set_src_dst(const coord_def &src, const coord_def &dst);
void set_annotate_map(bool annotate);
void set_distance_grid(travel_distance_grid_t distgrid);
void set_feature_vector(std::vector<coord_def> *features);
const coord_def travel_move() const;
const coord_def explore_target() const;
const coord_def greedy_square() const;
const coord_def unexplored_square() const;
protected:
bool is_greed_inducing_square(const coord_def &c) const;
bool path_examine_point(const coord_def &c);
virtual bool point_traverse_delay(const coord_def &c);
virtual bool path_flood(const coord_def &c, const coord_def &dc);
bool square_slows_movement(const coord_def &c);
void check_square_greed(const coord_def &c);
void good_square(const coord_def &c);
protected:
static const int UNFOUND_DIST = -30000;
static const int INFINITE_DIST = INFINITE_DISTANCE;
protected:
run_mode_type runmode;
coord_def start, dest;
coord_def next_travel_move;
bool floodout, double_flood;
bool ignore_hostile;
bool annotate_map;
const LevelStashes *ls;
bool need_for_greed;
coord_def unexplored_place, greedy_place;
int unexplored_dist, greedy_dist;
const int *refdist;
std::vector<coord_def> reseed_points;
std::vector<coord_def> *features;
travel_distance_col *point_distance;
int points;
int next_iter_points;
int traveled_distance;
int circ_index;
static FixedVector<coord_def, GXM * GYM> circumference[2];
};
extern TravelCache travel_cache;
#endif