The Battle for Wesnoth  1.19.10+dev
function_table.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2009 - 2025
3  by Bartosz Waresiak <dragonking@o2.pl>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
16 #include <queue>
17 #include <set>
18 #include <utility>
19 #include <vector>
20 
21 #include "ai/formula/ai.hpp"
24 
25 #include "ai/default/contexts.hpp"
26 
28 #include "attack_prediction.hpp"
29 #include "filesystem.hpp"
30 #include "game_board.hpp"
31 #include "display.hpp"
32 #include "log.hpp"
33 #include "map/label.hpp"
34 #include "map/map.hpp"
35 #include "pathfind/teleport.hpp"
36 #include "replay.hpp"
37 #include "resources.hpp"
38 #include "color.hpp"
39 #include "terrain/filter.hpp"
40 #include "units/unit.hpp"
41 #include "units/types.hpp"
42 #include "pathfind/pathfind.hpp"
43 
44 static lg::log_domain log_formula_ai("ai/engine/fai");
45 #define LOG_AI LOG_STREAM(info, log_formula_ai)
46 #define WRN_AI LOG_STREAM(warn, log_formula_ai)
47 #define ERR_AI LOG_STREAM(err, log_formula_ai)
48 
49 namespace wfl {
50 using ai::formula_ai;
51 
52 namespace {
53 
54 /*
55  * unit adapters let us treat unit and unit_type the same if we want to get access to attacks or movement cost
56  */
57 class unit_adapter {
58  public:
59  unit_adapter(const variant& arg) : unit_type_(), unit_() {
60  auto unit = arg.try_convert<unit_callable>();
61 
62  if (unit) {
63  unit_ = &unit->get_unit();
64  } else {
65  unit_type_ = &(arg.convert_to<unit_type_callable>()->get_unit_type());
66  }
67  }
68 
69  /**
70  * Estimates the damage this unit or unit_type would take from a single strike of an attack.
71  *
72  * Many details aren't taken into account here, for example abilities that are only active
73  * on offense, or on particular terrain. For unit_type, abilities aren't considered at all.
74  */
75  int damage_from(const attack_type& attack) const {
76  if(unit_type_ != nullptr) {
77  std::string type = attack.effective_damage_type().first;
79  return res;
80  } else {
81  return unit_->damage_from(attack, false, map_location());
82  }
83  }
84 
85  const_attack_itors attacks() const {
86  if(unit_type_ != nullptr) {
87  return unit_type_->attacks();
88  } else {
89  return unit_->attacks();
90  }
91  }
92 
93  int movement_cost(const t_translation::terrain_code & terrain) const {
94  if(unit_type_ != nullptr) {
95  return unit_type_->movement_type().movement_cost(terrain);
96  } else {
97  return unit_->movement_cost(terrain);
98  }
99  }
100 
101  private:
103  const unit* unit_;
104 };
105 
106 #define DEFINE_FAI_FUNCTION(name, min_args, max_args) \
107  class name##_function : public function_expression \
108  { \
109  public: \
110  explicit name##_function(const args_list& args, const formula_ai& ai) \
111  : function_expression(#name, args, min_args, max_args), ai_(ai) \
112  { \
113  } \
114  \
115  private: \
116  const formula_ai& ai_; \
117  variant execute(const formula_callable& variables, formula_debugger* fdb) const; \
118  }; \
119  \
120  variant name##_function::execute(const formula_callable& variables, formula_debugger* fdb) const
121 
122 DEFINE_FAI_FUNCTION(distance_to_nearest_unowned_village, 1, 1)
123 {
124  const map_location loc = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "distance_to_nearest_unowned_village:location")).convert_to<location_callable>()->loc();
125  int best = 1000000;
126  const std::vector<map_location>& villages = resources::gameboard->map().villages();
127  const std::set<map_location>& my_villages = ai_.current_team().villages();
128  for(std::vector<map_location>::const_iterator i = villages.begin(); i != villages.end(); ++i) {
129  int distance = distance_between(loc, *i);
130  if(distance < best) {
131  if(my_villages.count(*i) == 0) {
132  best = distance;
133  }
134  }
135  }
136 
137  return variant(best);
138 }
139 
140 static unsigned search_counter;
141 
142 namespace {
143  struct indexer {
144  int w, h;
145  indexer(int a, int b) : w(a), h(b) { }
146  int operator()(const map_location& loc) const {
147  return loc.y * w + loc.x;
148  }
149  };
150 
151  struct node {
154 
155  /**
156  * If equal to search_counter, the node is off the list.
157  * If equal to search_counter + 1, the node is on the list.
158  * Otherwise it is outdated.
159  */
160  unsigned in;
161 
162  node(int moves, const map_location &loc)
163  : movement_cost_(moves)
164  , loc_(loc)
165  , in(0)
166  {
167  }
168 
169  node()
170  : movement_cost_(0)
171  , loc_()
172  , in(0)
173  {
174  }
175 
176  bool operator<(const node& o) const {
177  return movement_cost_ < o.movement_cost_;
178  }
179  };
180 
181  struct comp {
182  const std::vector<node>& nodes;
183  comp(const std::vector<node>& n) : nodes(n) { }
184  bool operator()(int l, int r) const {
185  return nodes[r] < nodes[l];
186  }
187  };
188 
189  void find_movemap(const unit_adapter& u, const map_location& loc,
190  std::vector<int>& scores, bool allow_teleport, const formula_ai& ai_)
191  {
192  const std::set<map_location>& teleports = allow_teleport ? ai_.current_team().villages() : std::set<map_location>();
193 
194  const gamemap& map = resources::gameboard->map();
195 
196  std::vector<map_location> locs(6 + teleports.size());
197  std::copy(teleports.begin(), teleports.end(), locs.begin() + 6);
198 
199  search_counter += 2;
200  if (search_counter == 0) search_counter = 2;
201 
202  static std::vector<node> nodes;
203  nodes.resize(static_cast<size_t>(map.w()) * map.h());
204 
205  indexer index(map.w(), map.h());
206  comp node_comp(nodes);
207 
208  nodes[index(loc)] = node(0, loc);
209  std::vector<int> pq;
210  pq.push_back(index(loc));
211  while (!pq.empty()) {
212  node& n = nodes[pq.front()];
213  std::pop_heap(pq.begin(), pq.end(), node_comp);
214  pq.pop_back();
215  n.in = search_counter;
216 
217  get_adjacent_tiles(n.loc_, locs.data());
218  for (int i = teleports.count(n.loc_) ? locs.size() : 6; i-- > 0;) {
219  if (!locs[i].valid(map.w(), map.h())) continue;
220 
221  node& next = nodes[index(locs[i])];
222  bool next_visited = next.in - search_counter <= 1u;
223 
224  // test if the current path to locs[i] is better than this one could possibly be.
225  // we do this a couple more times below
226  if (next_visited && !(n < next)) continue;
227  const int move_cost = u.movement_cost(map[locs[i]]);
228 
229  node t = node(n.movement_cost_ + move_cost, locs[i]);
230 
231  if (next_visited && !(t < next)) continue;
232 
233  bool in_list = next.in == search_counter + 1;
234  t.in = search_counter + 1;
235  next = t;
236 
237  // if already in the priority queue then we just update it, else push it.
238  if (in_list) {
239  std::push_heap(pq.begin(), std::find(pq.begin(), pq.end(), index(locs[i])) + 1, node_comp);
240  }
241  else {
242  pq.push_back(index(locs[i]));
243  std::push_heap(pq.begin(), pq.end(), node_comp);
244  }
245  }
246  }
247 
248  for (int x = 0; x < map.w(); ++x) {
249  for (int y = 0; y < map.h(); ++y)
250  {
251  int i = y * map.w() + x;
252  const node &n = nodes[i];
253  scores[i] = scores[i] + n.movement_cost_;
254  //std::cout << x << "," << y << ":" << n.movement_cost << std::endl;
255  }
256  }
257  }
258 }
259 
260 DEFINE_FAI_FUNCTION(calculate_map_ownership, 2, 5)
261 {
262  int w = resources::gameboard->map().w();
263  int h = resources::gameboard->map().h();
264 
265  const variant units_input = args()[0]->evaluate(variables,fdb);
266  const variant leaders_input = args()[1]->evaluate(variables,fdb);
267 
268  int enemy_tolerance = 3;
269  if( args().size() > 2 )
270  enemy_tolerance = args()[2]->evaluate(variables,fdb).as_int();
271 
272  int enemy_border_tolerance = 5;
273  if( args().size() > 3 )
274  enemy_border_tolerance = args()[3]->evaluate(variables,fdb).as_int();
275 
276  int ally_tolerance = 3;
277  if( args().size() > 4 )
278  ally_tolerance = args()[4]->evaluate(variables,fdb).as_int();
279 
280  if( !units_input.is_list() )
281  return variant();
282 
283  std::size_t number_of_teams = units_input.num_elements();
284 
285  std::vector< std::vector<int>> scores( number_of_teams );
286 
287  for( std::size_t i = 0; i< number_of_teams; ++i)
288  scores[i].resize(static_cast<size_t>(w)*h);
289 
290  /* // TODO: Do we need this?
291  for(unit_map::const_iterator i = resources::gameboard->units().begin(); i != resources::gameboard->units().end(); ++i) {
292  unit_counter[i->second.side()-1]++;
293  unit_adapter unit(i->second);
294  find_movemap( resources::gameboard->map(), resources::gameboard->units(), unit, i->first, scores[i->second.side()-1], ai_.resources::gameboard->teams() , true );
295  }
296  */
297 
298  for(std::size_t side = 0 ; side < units_input.num_elements() ; ++side) {
299  if( leaders_input[side].is_empty() )
300  continue;
301 
302  const map_location loc = leaders_input[side][0].convert_to<location_callable>()->loc();
303  const variant units_of_side = units_input[side];
304 
305  for(std::size_t unit_it = 0 ; unit_it < units_of_side.num_elements() ; ++unit_it) {
306  unit_adapter unit(units_of_side[unit_it]);
307  find_movemap( unit, loc, scores[side], true, ai_ );
308  }
309  }
310 
311  std::size_t index = 0;
312  for( std::vector< std::vector<int>>::iterator i = scores.begin() ; i != scores.end() ; ++i) {
313  for( std::vector<int>::iterator j = i->begin() ; j != i->end() ; ++j ) {
314  if(units_input[index].num_elements() != 0) {
315  *j /= units_input[index].num_elements();
316  } else {
317  *j = 0;
318  }
319  }
320 
321  ++index;
322  }
323  //std::vector<variant> res;
324  std::map<variant, variant> res;
325 
326  std::size_t current_side = ai_.get_side() - 1 ;
327 
328  std::vector<int> enemies;
329  std::vector<int> allies;
330 
331  for(std::size_t side = 0 ; side < units_input.num_elements() ; ++side) {
332  if( side == current_side)
333  continue;
334 
335  if( ai_.current_team().is_enemy(side+1) ) {
336  if( !leaders_input[side].is_empty() )
337  enemies.push_back(side);
338  } else {
339  if( !leaders_input[side].is_empty() )
340  allies.push_back(side);
341  }
342  }
343 
344  //calculate_map_ownership( recruits_of_side, map(units_of_side, 'units', map( filter(units, leader), loc) ) )
345  //map(, debug_label(key,value))
346  for (int x = 0; x < w; ++x) {
347  for (int y = 0; y < h; ++y)
348  {
349  int i = y * w + x;
350  bool valid = true;
351  bool enemy_border = false;
352 
353  if( scores[current_side][i] > 98 )
354  continue;
355 
356  for (int side : enemies) {
357  int diff = scores[current_side][i] - scores[side][i];
358  if ( diff > enemy_tolerance) {
359  valid = false;
360  break;
361  } else if( std::abs(diff) < enemy_border_tolerance )
362  enemy_border = true;
363  }
364 
365  if( valid ) {
366  for (int side : allies) {
367  if ( scores[current_side][i] - scores[side][i] > ally_tolerance ) {
368  valid = false;
369  break;
370  }
371  }
372  }
373 
374  if( valid ) {
375  if( enemy_border )
376  res.emplace(variant(std::make_shared<location_callable>(map_location(x, y))), variant(scores[0][i] + 10000));
377  else
378  res.emplace(variant(std::make_shared<location_callable>(map_location(x, y))), variant(scores[0][i]));
379  }
380  }
381  }
382  return variant(res);
383 }
384 
385 DEFINE_WFL_FUNCTION(nearest_loc, 2, 2)
386 {
387  const map_location loc = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "nearest_loc:location")).convert_to<location_callable>()->loc();
388  variant items = args()[1]->evaluate(variables,add_debug_info(fdb,1,"nearest_loc:locations"));
389  int best = 1000000;
390  int best_i = -1;
391 
392  for(std::size_t i = 0; i < items.num_elements(); ++i) {
393 
394  const map_location move_loc = items[i].convert_to<location_callable>()->loc();
395  int distance = distance_between(loc, move_loc);
396 
397  if(distance < best) {
398  best = distance;
399  best_i = i;
400  }
401  }
402 
403  if( best_i != -1)
404  return variant(std::make_shared<location_callable>(items[best_i].convert_to<location_callable>()->loc()));
405  else
406  return variant();
407 }
408 
409 /** FormulaAI function to run fai script from file. Usable from in-game console.
410 * arguments[0] - required file name, follows the usual wml convention
411 */
412 DEFINE_FAI_FUNCTION(run_file, 1, 1)
413 {
414  const args_list& arguments = args();
415  const variant var0 = arguments[0]->evaluate(variables,add_debug_info(fdb,0,"run_file:file"));
416  const std::string filename = var0.string_cast();
417 
418  //NOTE: get_wml_location also filters file path to ensure it doesn't contain things like "../../top/secret"
420  if(!path) {
421  ERR_AI << "run_file : not found [" << filename <<"]";
422  return variant(); //no suitable file
423  }
424 
425  std::string formula_string = filesystem::read_file(path.value());
426  //need to get function_table from somewhere or delegate to someone who has access to it
427  formula_ptr parsed_formula = ai_.create_optional_formula(formula_string);
428  if(parsed_formula == formula_ptr()) {
429  ERR_AI << "run_file : unable to create formula";
430  return variant(); //was unable to create a formula from file
431  }
432  return parsed_formula->evaluate(variables,add_debug_info(fdb,-1,"run_file:formula_from_file"));
433 }
434 
435 DEFINE_WFL_FUNCTION(castle_locs, 1, 1)
436 {
437  const map_location starting_loc = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "castle_locs:location")).convert_to<location_callable>()->loc();
438 
439  //looks like reimplementing a generic graph search algorithm to me
440  std::set< map_location > visited_locs;
441  std::queue< map_location > queued_locs;
442 
443  queued_locs.push(starting_loc);
444 
445  while( !queued_locs.empty() ) {
446  const map_location loc = queued_locs.front();
447  queued_locs.pop();
448 
449  if ( visited_locs.find( loc ) != visited_locs.end() )
450  continue;
451 
452  visited_locs.insert(loc);
453 
454  for(const map_location& adj : get_adjacent_tiles(loc)) {
455  if (resources::gameboard->map().on_board(adj) && visited_locs.find( adj ) == visited_locs.end() ) {
456  if (resources::gameboard->map().get_terrain_info(adj).is_keep() ||
457  resources::gameboard->map().get_terrain_info(adj).is_castle() ) {
458  queued_locs.push(adj);
459  }
460  }
461  }
462  }
463 
464  if ( !resources::gameboard->map().get_terrain_info(starting_loc).is_keep() &&
465  !resources::gameboard->map().get_terrain_info(starting_loc).is_castle() )
466  visited_locs.erase(starting_loc);
467 
468  std::vector<variant> res;
469  for (const map_location& ml : visited_locs) {
470  res.emplace_back(std::make_shared<location_callable>( ml ));
471  }
472 
473  return variant(res);
474 }
475 
476 /**
477  * timeofday_modifer formula function. Returns combat modifier, taking
478  * alignment, illuminate, time of day and fearless trait into account.
479  * 'leadership' and 'slowed' are not taken into account.
480  * arguments[0] - unit
481  * arguments[1] - location (optional, defaults to unit's current location.
482  */
483 DEFINE_WFL_FUNCTION(timeofday_modifier, 1, 2)
484 {
485  variant u = args()[0]->evaluate(variables,add_debug_info(fdb,0,"timeofday_modifier:unit"));
486 
487  if( u.is_null() ) {
488  return variant();
489  }
490 
491  auto u_call = u.try_convert<unit_callable>();
492 
493  if(!u_call) {
494  return variant();
495  }
496 
497  const unit& un = u_call->get_unit();
498 
500 
501  if(args().size()==2) {
502  loc = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "timeofday_modifier:location")).convert_to<location_callable>()->loc();
503  }
504 
505  if(!loc.valid()) {
506  loc = u_call->get_location();
507  }
508 
509  return variant(combat_modifier(resources::gameboard->units(), resources::gameboard->map(), loc, un.alignment(), un.is_fearless()));
510 }
511 
512 DEFINE_FAI_FUNCTION(nearest_keep, 1, 1)
513 {
514  const map_location loc = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "nearest_keep:location")).convert_to<location_callable>()->loc();
515  int best = 1000000;
516  int best_i = -1;
517 
518  ai_.get_keeps();
520 
521  for( int i = 0 ; i < size; ++i) {
522  int distance = distance_between(loc, ai_.get_keeps_cache()[i].convert_to<location_callable>()->loc() );
523  if(distance < best)
524  {
525  best = distance;
526  best_i = i;
527  }
528  }
529 
530  if( best_i != -1)
531  return variant(std::make_shared<location_callable>(ai_.get_keeps_cache()[best_i].convert_to<location_callable>()->loc()));
532  else
533  return variant();
534 }
535 
536 /**
537 * Find suitable keep for unit at location
538 * arguments[0] - location for unit on which the suitable keep is to be found
539 */
540 DEFINE_FAI_FUNCTION(suitable_keep, 1, 1)
541 {
542  const map_location loc = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "suitable_keep:location")).convert_to<location_callable>()->loc();
543  const unit_map& units = resources::gameboard->units();
544  const unit_map::const_iterator u = units.find(loc);
545  if (u == units.end()){
546  return variant();
547  }
548  const pathfind::paths unit_paths(*u, false, true, ai_.current_team());
549  return variant(std::make_shared<location_callable>(ai_.suitable_keep(loc,unit_paths)));
550 }
551 
552 DEFINE_FAI_FUNCTION(find_shroud, 0, 1)
553 {
554  std::vector<variant> vars;
555  int w,h;
556 
557  if(args().size()==1) {
558  const gamemap& m = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "find_shroud:gamemap")).convert_to<gamemap_callable>()->get_gamemap();
559  w = m.w();
560  h = m.h();
561  } else {
562  w = resources::gameboard->map().w();
563  h = resources::gameboard->map().h();
564  }
565 
566  for(int i = 0; i < w; ++i)
567  for(int j = 0; j < h; ++j) {
569  vars.emplace_back(std::make_shared<location_callable>(map_location(i, j)));
570  }
571 
572  return variant(vars);
573 }
574 
575 DEFINE_FAI_FUNCTION(close_enemies, 2, 2)
576 {
577  std::vector<variant> vars;
578  const map_location loc = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "close_enemies:location")).convert_to<location_callable>()->loc();
579  int range_s = args()[1]->evaluate(variables,add_debug_info(fdb,1,"close_enemies:distance")).as_int();
580  if (range_s < 0) {
581  WRN_AI << "close_enemies_function: range is negative (" << range_s << ")";
582  range_s = 0;
583  }
584  std::size_t range = static_cast<std::size_t>(range_s);
587  while (un != end) {
588  if (distance_between(loc, un->get_location()) <= range) {
589  if (un->side() != ai_.get_side()) {//fixme: ignores allied units
590  vars.emplace_back(std::make_shared<unit_callable>(*un));
591  }
592  }
593  ++un;
594  }
595  return variant(vars);
596 }
597 
598 DEFINE_WFL_FUNCTION(calculate_outcome, 3, 4)
599 {
600  std::vector<variant> vars;
601  int weapon;
602  if (args().size() > 3) weapon = args()[3]->evaluate(variables,add_debug_info(fdb,3,"calculate_outcome:weapon")).as_int();
603  else weapon = -1;
604 
605  const unit_map& units = resources::gameboard->units();
606  map_location attacker_location =
607  args()[0]->evaluate(variables, add_debug_info(fdb, 0, "calculate_outcome:attacker_current_location")).convert_to<location_callable>()->loc();
608  if(units.count(attacker_location) == 0) {
609  ERR_AI << "Performing calculate_outcome() with non-existent attacker at (" <<
610  attacker_location.wml_x() << "," << attacker_location.wml_y() << ")";
611  return variant();
612  }
613 
614  map_location defender_location =
615  args()[2]->evaluate(variables,add_debug_info(fdb, 2, "calculate_outcome:defender_location")).convert_to<location_callable>()->loc();
616  if(units.count(defender_location) == 0) {
617  ERR_AI << "Performing calculate_outcome() with non-existent defender at (" <<
618  defender_location.wml_x() << "," << defender_location.wml_y() << ")";
619  return variant();
620  }
621 
622  battle_context bc(units, args()[1]->evaluate(variables, add_debug_info(fdb, 1, "calculate_outcome:attacker_attack_location")).convert_to<location_callable>()->loc(),
623  defender_location, weapon, -1, 1.0, nullptr, units.find(attacker_location).get_shared_ptr());
624  std::vector<double> hp_dist = bc.get_attacker_combatant().hp_dist;
625  std::vector<double>::iterator it = hp_dist.begin();
626  int i = 0;
627  std::vector<variant> hitLeft;
628  std::vector<variant> prob;
629  while (it != hp_dist.end()) {
630  if (*it != 0) {
631  hitLeft.emplace_back(i);
632  prob.emplace_back(static_cast<int>(*it*10000));
633  }
634  ++it;
635  ++i;
636  }
637  std::vector<variant> status;
638  if (bc.get_attacker_combatant().poisoned != 0)
639  status.emplace_back("Poisoned");
640  if (bc.get_attacker_combatant().slowed != 0)
641  status.emplace_back("Slowed");
642  if (bc.get_defender_stats().petrifies && static_cast<unsigned int>(hitLeft[0].as_int()) != bc.get_attacker_stats().hp)
643  status.emplace_back("Stoned");
644  if (bc.get_defender_stats().plagues && hitLeft[0].as_int() == 0)
645  status.emplace_back("Zombiefied");
646  vars.emplace_back(std::make_shared<outcome_callable>(hitLeft, prob, status));
647  hitLeft.clear();
648  prob.clear();
649  status.clear();
650  hp_dist = bc.get_defender_combatant().hp_dist;
651  it = hp_dist.begin();
652  i = 0;
653  while (it != hp_dist.end()) {
654  if (*it != 0) {
655  hitLeft.emplace_back(i);
656  prob.emplace_back(static_cast<int>(*it*10000));
657  }
658  ++it;
659  ++i;
660  }
661  if (bc.get_defender_combatant().poisoned != 0)
662  status.emplace_back("Poisoned");
663  if (bc.get_defender_combatant().slowed != 0)
664  status.emplace_back("Slowed");
665  if (bc.get_attacker_stats().petrifies && static_cast<unsigned int>(hitLeft[0].as_int()) != bc.get_defender_stats().hp)
666  status.emplace_back("Stoned");
667  if (bc.get_attacker_stats().plagues && hitLeft[0].as_int() == 0)
668  status.emplace_back("Zombiefied");
669  vars.emplace_back(std::make_shared<outcome_callable>(hitLeft, prob, status));
670  return variant(vars);
671 }
672 
673 DEFINE_WFL_FUNCTION(outcomes, 1, 1)
674 {
675  variant attack = args()[0]->evaluate(variables,add_debug_info(fdb,0,"outcomes:attack"));
676  auto analysis = attack.convert_to<ai::attack_analysis>();
677  //unit_map units_with_moves(resources::gameboard->units());
678  //typedef std::pair<map_location, map_location> mv;
679  //for(const mv &m : analysis->movements) {
680  // units_with_moves.move(m.first, m.second);
681  //}
682 
683  std::vector<variant> vars;
684  if(analysis->chance_to_kill > 0.0) {
685  //unit_map units(units_with_moves);
686  //units.erase(analysis->target);
687  vars.emplace_back(std::make_shared<position_callable>(/*&units,*/ static_cast<int>(analysis->chance_to_kill*100)));
688 
689  }
690 
691  if(analysis->chance_to_kill < 1.0) {
692  //unit_map units(units_with_moves);
693  vars.emplace_back(std::make_shared<position_callable>(/*&units,*/ static_cast<int>(100 - analysis->chance_to_kill*100)));
694  }
695 
696  return variant(vars);
697 }
698 
699 DEFINE_FAI_FUNCTION(rate_action, 1, 1)
700 {
701  variant act = args()[0]->evaluate(variables,add_debug_info(fdb,0,"rate_action:action"));
702  auto analysis = act.convert_to<ai::attack_analysis>();
703 
704  return variant(analysis->rating(ai_.get_aggression(),ai_)*1000,variant::DECIMAL_VARIANT);
705 }
706 
707 DEFINE_WFL_FUNCTION(recall, 1, 2)
708 {
709  const std::string id = args()[0]->evaluate(variables,add_debug_info(fdb,0,"recall:id")).as_string();
711  if(args().size() >= 2) {
712  loc = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "recall:location")).convert_to<location_callable>()->loc();
713  }
714 
715  return variant(std::make_shared<recall_callable>(loc, id));
716 }
717 
718 DEFINE_WFL_FUNCTION(recruit, 1, 2)
719 {
720  const std::string type = args()[0]->evaluate(variables,add_debug_info(fdb,0,"recruit:type")).as_string();
722  if(args().size() >= 2) {
723  loc = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "recruit:location")).convert_to<location_callable>()->loc();
724  }
725 
726  return variant(std::make_shared<recruit_callable>(loc, type));
727 }
728 
729 DEFINE_FAI_FUNCTION(shortest_path, 2, 3)
730 {
731 
732  std::vector<variant> locations;
733 
734  const map_location src = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "shortest_path:src")).convert_to<location_callable>()->loc();
735  const map_location dst = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "shortest_path:dst")).convert_to<location_callable>()->loc();
736  map_location unit_loc;
737 
738  if( src == dst )
739  return variant(locations);
740 
741  if(args().size() > 2)
742  unit_loc = args()[2]->evaluate(variables,add_debug_info(fdb,2,"shortest_path:unit_location")).convert_to<location_callable>()->loc();
743  else
744  unit_loc = src;
745 
746  unit_map::iterator unit_it = resources::gameboard->units().find(unit_loc);
747 
748  if( unit_it == resources::gameboard->units().end() ) {
749  std::ostringstream str;
750  str << "shortest_path function: expected unit at location (" << (unit_loc.wml_x()) << "," << (unit_loc.wml_y()) << ")";
751  throw formula_error( str.str(), "", "", 0);
752  }
753 
754  pathfind::teleport_map allowed_teleports = ai_.get_allowed_teleports(unit_it);
755 
756  pathfind::plain_route route = ai_.shortest_path_calculator( src, dst, unit_it, allowed_teleports );
757 
758  if( route.steps.size() < 2 ) {
759  return variant(locations);
760  }
761 
762  for (std::vector<map_location>::const_iterator loc_iter = route.steps.begin() + 1 ; loc_iter !=route.steps.end(); ++loc_iter) {
763  locations.emplace_back(std::make_shared<location_callable>(*loc_iter));
764  }
765 
766  return variant(locations);
767 }
768 
769 DEFINE_FAI_FUNCTION(simplest_path, 2, 3)
770 {
771  std::vector<variant> locations;
772 
773  const map_location src = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "simplest_path:src")).convert_to<location_callable>()->loc();
774  const map_location dst = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "simplest_path:dst")).convert_to<location_callable>()->loc();
775  map_location unit_loc;
776 
777  if( src == dst )
778  return variant(locations);
779 
780  if(args().size() > 2)
781  unit_loc = args()[2]->evaluate(variables, add_debug_info(fdb, 2, "simplest_path:unit_location")).convert_to<location_callable>()->loc();
782  else
783  unit_loc = src;
784 
785  unit_map::iterator unit_it = resources::gameboard->units().find(unit_loc);
786 
787  if( unit_it == resources::gameboard->units().end() ) {
788  std::ostringstream str;
789  str << "simplest_path function: expected unit at location (" << (unit_loc.wml_x()) << "," << (unit_loc.wml_y()) << ")";
790  throw formula_error( str.str(), "", "", 0);
791  }
792 
793  pathfind::teleport_map allowed_teleports = ai_.get_allowed_teleports(unit_it);
794 
796 
797  pathfind::plain_route route = pathfind::a_star_search(src, dst, 1000.0, em_calc, resources::gameboard->map().w(), resources::gameboard->map().h(), &allowed_teleports);
798 
799  if( route.steps.size() < 2 ) {
800  return variant(locations);
801  }
802 
803  for (std::vector<map_location>::const_iterator loc_iter = route.steps.begin() + 1 ; loc_iter !=route.steps.end(); ++loc_iter) {
804  if(unit_it->movement_cost(static_cast<const game_board*>(resources::gameboard)->map()[*loc_iter]) < movetype::UNREACHABLE) {
805  locations.emplace_back(std::make_shared<location_callable>(*loc_iter));
806  } else {
807  break;
808  }
809  }
810 
811  return variant(locations);
812 }
813 
814 DEFINE_FAI_FUNCTION(next_hop, 2, 3)
815 {
816 
817  const map_location src = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "next_hop:src")).convert_to<location_callable>()->loc();
818  const map_location dst = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "next_hop:dst")).convert_to<location_callable>()->loc();
819  map_location unit_loc;
820 
821  if( src == dst )
822  return variant();
823 
824  if(args().size() > 2)
825  unit_loc = args()[2]->evaluate(variables, add_debug_info(fdb, 2, "next_hop:unit_location")).convert_to<location_callable>()->loc();
826  else
827  unit_loc = src;
828 
829  unit_map::iterator unit_it = resources::gameboard->units().find(unit_loc);
830 
831  if( unit_it == resources::gameboard->units().end() ) {
832  std::ostringstream str;
833  str << "next_hop function: expected unit at location (" << (unit_loc.wml_x()) << "," << (unit_loc.wml_y()) << ")";
834  throw formula_error( str.str(), "", "", 0);
835  }
836 
837  pathfind::teleport_map allowed_teleports = ai_.get_allowed_teleports(unit_it);
838 
839  pathfind::plain_route route = ai_.shortest_path_calculator( src, dst, unit_it, allowed_teleports );
840 
841  if( route.steps.size() < 2 ) {
842  return variant();
843  }
844 
846  const ai::moves_map &possible_moves = ai_.get_possible_moves();
847  const ai::moves_map::const_iterator& p_it = possible_moves.find(unit_loc);
848  if (p_it==possible_moves.end() ) {
849  return variant();
850  }
851 
852  for (std::vector<map_location>::const_iterator loc_iter = route.steps.begin() + 1 ; loc_iter !=route.steps.end(); ++loc_iter) {
853 
854  if (p_it->second.destinations.find(*loc_iter) != p_it->second.destinations.end() ) {
855  loc = *loc_iter;
856  } else {
857  break;
858  }
859  }
861  return variant();
862  }
863  return variant(std::make_shared<location_callable>(loc));
864 }
865 
866 DEFINE_WFL_FUNCTION(move, 2, 2)
867 {
868  const map_location src = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "move:src")).convert_to<location_callable>()->loc();
869  const map_location dst = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "move:dst")).convert_to<location_callable>()->loc();
870  LOG_AI << "move(): " << src << ", " << dst << ")";
871  return variant(std::make_shared<move_callable>(src, dst));
872 }
873 
874 DEFINE_WFL_FUNCTION(move_partial, 2, 2)
875 {
876  const map_location src = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "move_partial:src")).convert_to<location_callable>()->loc();
877  const map_location dst = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "move_partial:dst")).convert_to<location_callable>()->loc();
878  LOG_AI << "move_partial(): " << src << ", " << dst << ")";
879  return variant(std::make_shared<move_partial_callable>(src, dst));
880 }
881 
882 DEFINE_WFL_FUNCTION(set_unit_var, 3, 3)
883 {
884  return variant(std::make_shared<set_unit_var_callable>(args()[0]->evaluate(variables,add_debug_info(fdb,0,"set_unit_var:key")).as_string(), args()[1]->evaluate(variables,add_debug_info(fdb,1,"set_unit_var:value")), args()[2]->evaluate(variables,add_debug_info(fdb,2,"set_unit_var:unit_location")).convert_to<location_callable>()->loc()));
885 }
886 
887 DEFINE_WFL_FUNCTION(fallback, 0, 1)
888 {
889  // The parameter is not used, but is accepted for legacy compatibility
890  if(args().size() == 1 && args()[0]->evaluate(variables).as_string() != "human")
891  return variant();
892  return variant(std::make_shared<fallback_callable>());
893 }
894 
895 DEFINE_WFL_FUNCTION(attack, 3, 4)
896 {
897  const map_location move_from = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "attack:move_from")).convert_to<location_callable>()->loc();
898  const map_location src = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "attack:src")).convert_to<location_callable>()->loc();
899  const map_location dst = args()[2]->evaluate(variables, add_debug_info(fdb, 2, "attack:dst")).convert_to<location_callable>()->loc();
900  const int weapon = args().size() == 4 ? args()[3]->evaluate(variables,add_debug_info(fdb,3,"attack:weapon")).as_int() : -1;
901  if(resources::gameboard->units().count(move_from) == 0 || resources::gameboard->units().count(dst) == 0) {
902  ERR_AI << "AI ERROR: Formula produced illegal attack: " << move_from << " -> " << src << " -> " << dst;
903  return variant();
904  }
905  return variant(std::make_shared<attack_callable>(move_from, src, dst, weapon));
906 }
907 
908 DEFINE_FAI_FUNCTION(debug_label, 2, 2)
909 {
910  const args_list& arguments = args();
911  const variant var0 = arguments[0]->evaluate(variables,fdb);
912  const variant var1 = arguments[1]->evaluate(variables,fdb);
913 
914  const map_location location = var0.convert_to<location_callable>()->loc();
915  std::string text;
916  if( var1.is_string() )
917  text = var1.as_string();
918  else
919  text = var1.to_debug_string();
920 
922  std::string team_name;
923 
925 
926  const terrain_label *res;
927  res = gui->labels().set_label(location, text, ai_.get_side() - 1, team_name, color);
928  if (res && resources::recorder)
930 
931  std::vector<variant> result;
932  result.push_back(var0);
933  result.push_back(var1);
934  return variant(result);
935 }
936 
937 DEFINE_WFL_FUNCTION(is_village, 2, 3)
938 {
939  const gamemap& m = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "is_village:map")).convert_to<gamemap_callable>()->get_gamemap();
940 
942  if(args().size() == 2) {
943  loc = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "is_village:location")).convert_to<location_callable>()->loc();
944  } else {
945  loc = map_location( args()[1]->evaluate(variables,add_debug_info(fdb,1,"is_village:x")).as_int(),
946  args()[2]->evaluate(variables,add_debug_info(fdb,2,"is_village:y")).as_int(), wml_loc());
947  }
948  return variant(m.is_village(loc));
949 }
950 
951 DEFINE_FAI_FUNCTION(is_unowned_village, 2, 3)
952 {
953 
954  const gamemap& m = args()[0]->evaluate(variables, add_debug_info(fdb, 0, "is_unowned_village:map")).convert_to<gamemap_callable>()->get_gamemap();
955  const std::set<map_location>& my_villages = ai_.current_team().villages();
956 
958  if(args().size() == 2) {
959  loc = args()[1]->evaluate(variables, add_debug_info(fdb, 1, "is_unowned_village:location")).convert_to<location_callable>()->loc();
960  } else {
961  loc = map_location( args()[1]->evaluate(variables,add_debug_info(fdb,1,"is_unowned_village:x")).as_int(),
962  args()[2]->evaluate(variables,add_debug_info(fdb,2,"is_unowned_village:y")).as_int(), wml_loc());
963  }
964 
965  if(m.is_village(loc) && (my_villages.count(loc)==0) ) {
966  return variant(true);
967  } else {
968  return variant(false);
969  }
970 }
971 
973 {
974  variant res = args()[0]->evaluate(variables,add_debug_info(fdb,0,"unit_moves:unit_location"));
975  std::vector<variant> vars;
976  if(res.is_null()) {
977  return variant(vars);
978  }
979 
980  const map_location& loc = res.convert_to<location_callable>()->loc();
981  const ai::move_map& srcdst = ai_.get_srcdst();
982  typedef ai::move_map::const_iterator Itor;
983  std::pair<Itor,Itor> range = srcdst.equal_range(loc);
984 
985  for(Itor i = range.first; i != range.second; ++i) {
986  vars.emplace_back(std::make_shared<location_callable>(i->second));
987  }
988 
989  return variant(vars);
990 }
991 
992 DEFINE_WFL_FUNCTION(units_can_reach, 2, 2)
993 {
994  std::vector<variant> vars;
995  variant dstsrc_var = args()[0]->evaluate(variables,add_debug_info(fdb,0,"units_can_reach:possible_move_list"));
996  const ai::move_map& dstsrc = dstsrc_var.convert_to<move_map_callable>()->dstsrc();
997  std::pair<ai::move_map::const_iterator,ai::move_map::const_iterator> range =
998  dstsrc.equal_range(args()[1]->evaluate(variables, add_debug_info(fdb, 1, "units_can_reach:possible_move_list")).convert_to<location_callable>()->loc());
999  while(range.first != range.second) {
1000  unit_map::const_iterator un = resources::gameboard->units().find(range.first->second);
1001  assert(un != resources::gameboard->units().end());
1002  vars.emplace_back(std::make_shared<unit_callable>(*un));
1003  ++range.first;
1004  }
1005 
1006  return variant(vars);
1007 }
1008 
1009 DEFINE_FAI_FUNCTION(is_avoided_location, 1, 1)
1010 {
1011  variant res = args()[0]->evaluate(variables,add_debug_info(fdb,0,"is_avoided_location:location"));
1012  if(res.is_null()) {
1013  return variant();
1014  }
1015  const map_location& loc = res.convert_to<location_callable>()->loc();
1016  return variant(ai_.get_avoid().match(loc));
1017 }
1018 
1019 DEFINE_WFL_FUNCTION(max_possible_damage, 2, 2)
1020 {
1021  variant u1 = args()[0]->evaluate(variables,add_debug_info(fdb,0,"max_possible_damage:unit1"));
1022  variant u2 = args()[1]->evaluate(variables,add_debug_info(fdb,1,"max_possible_damage:unit2"));
1023  if(u1.is_null() || u2.is_null()) {
1024  return variant();
1025  }
1026 
1027  unit_adapter u_attacker(u1), u_defender(u2);
1028  int best = 0;
1029  for(const attack_type& atk : u_attacker.attacks()) {
1030  const int dmg = round_damage(atk.damage(), u_defender.damage_from(atk), 100) * atk.num_attacks();
1031  if(dmg > best)
1032  best = dmg;
1033  }
1034  return variant(best);
1035 }
1036 
1037 namespace {
1038  std::pair<int, int> best_melee_and_ranged_attacks(unit_adapter attacker, unit_adapter defender) {
1039  int highest_melee_damage = 0;
1040  int highest_ranged_damage = 0;
1041 
1042  for (const attack_type &attack : attacker.attacks()) {
1043  const int dmg = round_damage(attack.damage(), defender.damage_from(attack), 100) * attack.num_attacks();
1044  if (attack.range() == "melee") {
1045  highest_melee_damage = std::max(highest_melee_damage, dmg);
1046  } else {
1047  highest_ranged_damage = std::max(highest_ranged_damage, dmg);
1048  }
1049  }
1050 
1051  return std::pair(highest_melee_damage, highest_ranged_damage);
1052  }
1053 }
1054 
1055 DEFINE_WFL_FUNCTION(max_possible_damage_with_retaliation, 2, 2)
1056 {
1057  variant u1 = args()[0]->evaluate(variables,add_debug_info(fdb,0,"max_possible_damage_with_retaliation:unit1"));
1058  variant u2 = args()[1]->evaluate(variables,add_debug_info(fdb,1,"max_possible_damage_with_retaliation:unit2"));
1059 
1060  if(u1.is_null() || u2.is_null()) {
1061  return variant();
1062  }
1063 
1064  unit_adapter attacker(u1);
1065  unit_adapter defender(u2);
1066 
1067  // find max damage inflicted by attacker and by defender to the attacker
1068  std::pair<int, int> best_attacker_attacks = best_melee_and_ranged_attacks(attacker, defender);
1069  std::pair<int, int> best_defender_attacks = best_melee_and_ranged_attacks(defender, attacker);
1070 
1071  std::vector<variant> vars;
1072  vars.emplace_back(best_attacker_attacks.first);
1073  vars.emplace_back(best_attacker_attacks.second);
1074  vars.emplace_back(best_defender_attacks.first);
1075  vars.emplace_back(best_defender_attacks.second);
1076 
1077  return variant(vars);
1078 }
1079 
1080 template<typename T>
1081 class ai_formula_function : public formula_function {
1082 protected:
1083  formula_ai& ai_;
1084 public:
1085  ai_formula_function(const std::string& name, ai::formula_ai& ai) : formula_function(name), ai_(ai) {}
1086  function_expression_ptr generate_function_expression(const std::vector<expression_ptr>& args) const {
1087  return std::make_shared<T>(args, ai_);
1088  }
1089 };
1090 
1091 }
1092 
1093 // This macro is for functions taking an additional formula_ai argument.
1094 // Functions using the other macro could potentially be made core.
1095 #define DECLARE_FAI_FUNCTION(name) \
1096  add_function(#name, std::make_shared<ai_formula_function<name##_function>>(#name, ai))
1097 
1100 {
1101  function_symbol_table& functions_table = *this;
1102  DECLARE_WFL_FUNCTION(outcomes);
1103  //DECLARE_FAI_FUNCTION(evaluate_for_position);
1104  DECLARE_WFL_FUNCTION(move);
1105  DECLARE_WFL_FUNCTION(move_partial);
1106  DECLARE_WFL_FUNCTION(attack);
1107  DECLARE_FAI_FUNCTION(rate_action);
1108  DECLARE_WFL_FUNCTION(recall);
1109  DECLARE_WFL_FUNCTION(recruit);
1110  DECLARE_FAI_FUNCTION(is_avoided_location);
1111  DECLARE_WFL_FUNCTION(is_village);
1112  DECLARE_FAI_FUNCTION(is_unowned_village);
1114  DECLARE_WFL_FUNCTION(set_unit_var);
1115  DECLARE_WFL_FUNCTION(fallback);
1116  DECLARE_WFL_FUNCTION(units_can_reach);
1117  DECLARE_FAI_FUNCTION(debug_label);
1118  DECLARE_WFL_FUNCTION(max_possible_damage);
1119  DECLARE_WFL_FUNCTION(max_possible_damage_with_retaliation);
1120  DECLARE_FAI_FUNCTION(next_hop);
1121  DECLARE_WFL_FUNCTION(castle_locs);
1122  DECLARE_WFL_FUNCTION(timeofday_modifier);
1123  DECLARE_FAI_FUNCTION(distance_to_nearest_unowned_village);
1124  DECLARE_FAI_FUNCTION(shortest_path);
1125  DECLARE_FAI_FUNCTION(simplest_path);
1126  DECLARE_FAI_FUNCTION(nearest_keep);
1127  DECLARE_FAI_FUNCTION(suitable_keep);
1128  DECLARE_WFL_FUNCTION(nearest_loc);
1129  DECLARE_FAI_FUNCTION(find_shroud);
1130  DECLARE_FAI_FUNCTION(close_enemies);
1131  DECLARE_WFL_FUNCTION(calculate_outcome);
1132  DECLARE_FAI_FUNCTION(run_file);
1133  DECLARE_FAI_FUNCTION(calculate_map_ownership);
1134 }
1135 #undef DECLARE_WFL_FUNCTION
1136 
1137 }
int combat_modifier(const unit_map &units, const gamemap &map, const map_location &loc, unit_alignments::type alignment, bool is_fearless)
Returns the amount that a unit's damage should be multiplied by due to the current time of day.
Definition: attack.cpp:1600
map_location loc
Definition: move.cpp:172
double t
Definition: astarsearch.cpp:63
boost::iterator_range< boost::indirect_iterator< attack_list::const_iterator > > const_attack_itors
const wfl::variant & get_keeps_cache() const
Definition: ai.hpp:114
pathfind::teleport_map get_allowed_teleports(unit_map::iterator &unit_it) const
Definition: ai.cpp:248
wfl::formula_ptr create_optional_formula(const std::string &formula_string) const
Create a new formula from the string, using the symbol table which is stored in the AI.
Definition: ai.cpp:133
wfl::variant get_keeps() const
Definition: ai.cpp:585
pathfind::plain_route shortest_path_calculator(const map_location &src, const map_location &dst, unit_map::iterator &unit_it, pathfind::teleport_map &allowed_teleports) const
Definition: ai.cpp:194
virtual const map_location & suitable_keep(const map_location &leader_location, const pathfind::paths &leader_paths) const override
get most suitable keep for leader - nearest free that can be reached in 1 turn, if none - return near...
Definition: contexts.hpp:856
virtual const team & current_team() const override
Definition: contexts.hpp:450
virtual const terrain_filter & get_avoid() const override
Definition: contexts.hpp:586
virtual const move_map & get_srcdst() const override
Definition: contexts.hpp:716
virtual double get_aggression() const override
Definition: contexts.hpp:546
virtual const moves_map & get_possible_moves() const override
Definition: contexts.hpp:676
virtual side_number get_side() const override
Get the side number.
Definition: contexts.hpp:396
std::pair< std::string, int > effective_damage_type() const
The type of attack used and the resistance value that does the most damage.
Definition: abilities.cpp:1325
Computes the statistics of a battle between an attacker and a defender unit.
Definition: attack.hpp:167
Sort-of-Singleton that many classes, both GUI and non-GUI, use to access the game data.
Definition: display.hpp:96
static display * get_singleton()
Returns the display object if a display object exists.
Definition: display.hpp:110
Game board class.
Definition: game_board.hpp:47
virtual const unit_map & units() const override
Definition: game_board.hpp:107
virtual const gamemap & map() const override
Definition: game_board.hpp:97
int w() const
Effective map width.
Definition: map.hpp:50
int h() const
Effective map height.
Definition: map.hpp:53
Encapsulates the map of the game.
Definition: map.hpp:172
bool is_village(const map_location &loc) const
Definition: map.cpp:66
const std::vector< map_location > & villages() const
Return a list of the locations of villages on the map.
Definition: map.hpp:237
static const int UNREACHABLE
Magic value that signifies a hex is unreachable.
Definition: movetype.hpp:174
int resistance_against(const std::string &damage_type) const
Returns the vulnerability to the indicated damage type (higher means takes more damage).
Definition: movetype.hpp:292
int movement_cost(const t_translation::terrain_code &terrain, bool slowed=false) const
Returns the cost to move through the indicated terrain.
Definition: movetype.hpp:278
void add_label(const terrain_label *)
Definition: replay.cpp:258
bool is_enemy(int n) const
Definition: team.hpp:234
const std::set< map_location > & villages() const
Definition: team.hpp:176
bool shrouded(const map_location &loc) const
Definition: team.cpp:640
static color_t get_side_color(int side)
Definition: team.cpp:949
To store label data Class implements logic for rendering.
Definition: label.hpp:111
Container associating units to locations.
Definition: map.hpp:98
unit_iterator end()
Definition: map.hpp:428
std::size_t count(const map_location &loc) const
Definition: map.hpp:413
unit_iterator find(std::size_t id)
Definition: map.cpp:302
unit_iterator begin()
Definition: map.hpp:418
A single unit type that the player may recruit.
Definition: types.hpp:43
const movetype & movement_type() const
Definition: types.hpp:189
const_attack_itors attacks() const
Definition: types.cpp:545
This class represents a single unit of a specific type.
Definition: unit.hpp:133
ai_function_symbol_table(ai::formula_ai &ai)
std::size_t num_elements() const
Definition: variant.cpp:267
@ DECIMAL_VARIANT
Definition: variant.hpp:31
std::shared_ptr< T > convert_to() const
Definition: variant.hpp:100
Default AI contexts.
map_display and display: classes which take care of displaying the map and game-data on the screen.
Declarations for File-IO.
Defines formula ai.
std::size_t i
Definition: function.cpp:1022
#define DEFINE_WFL_FUNCTION(name, min_args, max_args)
Helper macro to declare an associated class for a WFL function.
Definition: function.hpp:27
#define DECLARE_WFL_FUNCTION(name)
Declares a function name in the local function table functions_table.
Definition: function.hpp:47
#define ERR_AI
const unit * unit_
formula_ai & ai_
static lg::log_domain log_formula_ai("ai/engine/fai")
int h
map_location loc_
#define LOG_AI
#define WRN_AI
#define DEFINE_FAI_FUNCTION(name, min_args, max_args)
int movement_cost_
const unit_type * unit_type_
const std::vector< node > & nodes
int w
#define DECLARE_FAI_FUNCTION(name)
unsigned in
If equal to search_counter, the node is off the list.
static bool operator<(const placing_info &a, const placing_info &b)
Definition: game_state.cpp:106
unit_alignments::type alignment() const
The alignment of this unit.
Definition: unit.hpp:491
int damage_from(const attack_type &attack, bool attacker, const map_location &loc, const_attack_ptr weapon=nullptr) const
Calculates the damage this unit would take from a certain attack.
Definition: unit.hpp:987
attack_itors attacks()
Gets an iterator over this unit's attacks.
Definition: unit.hpp:942
int movement_cost(const t_translation::terrain_code &terrain) const
Get the unit's movement cost on a particular terrain.
Definition: unit.hpp:1496
bool is_fearless() const
Gets whether this unit is fearless - ie, unaffected by time of day.
Definition: unit.hpp:1303
T end(const std::pair< T, T > &p)
void get_adjacent_tiles(const map_location &a, map_location *res)
Function which, given a location, will place all adjacent locations in res.
Definition: location.cpp:479
std::size_t distance_between(const map_location &a, const map_location &b)
Function which gives the number of hexes between two tiles (i.e.
Definition: location.cpp:550
Standard logging facilities (interface).
constexpr int round_damage(double base_damage, int bonus, int divisor)
round (base_damage * bonus / divisor) to the closest integer, but up or down towards base_damage
Definition: math.hpp:80
A small explanation about what's going on here: Each action has access to two game_info objects First...
Definition: actions.cpp:59
std::multimap< map_location, map_location > move_map
The standard way in which a map of possible moves is recorded.
Definition: game_info.hpp:43
std::map< map_location, pathfind::paths > moves_map
The standard way in which a map of possible movement routes to location is recorded.
Definition: game_info.hpp:46
utils::optional< std::string > get_wml_location(const std::string &path, const utils::optional< std::string > &current_dir)
Returns a translated path to the actual file or directory, if it exists.
std::string read_file(const std::string &fname)
Basic disk I/O - read file.
std::string path
Definition: filesystem.cpp:93
General purpose widgets.
plain_route a_star_search(const map_location &src, const map_location &dst, double stop_at, const cost_calculator &calc, const std::size_t width, const std::size_t height, const teleport_map *teleports, bool border)
game_board * gameboard
Definition: resources.cpp:20
replay * recorder
Definition: resources.cpp:28
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
Definition: unicode.cpp:85
std::size_t index(std::string_view str, const std::size_t index)
Codepoint index corresponding to the nth character in a UTF-8 string.
Definition: unicode.cpp:70
auto * find(Container &container, const Value &value)
Convenience wrapper for using find on a container without needing to comare to end()
Definition: general.hpp:140
std::string::const_iterator iterator
Definition: tokenizer.hpp:25
Definition: contexts.hpp:43
formula_debugger * add_debug_info(formula_debugger *fdb, int arg_number, const std::string &f_name)
std::shared_ptr< formula > formula_ptr
Definition: formula_fwd.hpp:22
std::shared_ptr< function_expression > function_expression_ptr
Definition: function.hpp:172
This module contains various pathfinding functions and utilities.
Replay control code.
static config unit_moves(const reports::context &rc, const unit *u, bool is_visible_unit)
Definition: reports.cpp:664
rect dst
Location on the final composed sheet.
rect src
Non-transparent portion of the surface to compose.
std::string filename
Filename.
The basic class for representing 8-bit RGB or RGBA colour values.
Definition: color.hpp:59
Encapsulates the map of the game.
Definition: location.hpp:45
bool valid() const
Definition: location.hpp:110
int wml_y() const
Definition: location.hpp:218
static const map_location & null_location()
Definition: location.hpp:102
int wml_x() const
Definition: location.hpp:217
Function which only uses terrain, ignoring shroud, enemies, etc.
Definition: pathfind.hpp:242
Object which contains all the possible locations a unit can move to, with associated best routes to t...
Definition: pathfind.hpp:73
Structure which holds a single route between one location and another.
Definition: pathfind.hpp:133
std::vector< map_location > steps
Definition: pathfind.hpp:135
A terrain string which is converted to a terrain is a string with 1 or 2 layers the layers are separa...
Definition: translation.hpp:49
pointer get_shared_ptr() const
This is exactly the same as operator-> but it's slightly more readable, and can replace &*iter syntax...
Definition: map.hpp:217
static map_location::direction n
static const unit_type & get_unit_type(const std::string &type_id)
Converts a string ID to a unit_type.
Definition: unit.cpp:216
#define b