54 #define ERR_NG LOG_STREAM(err, log_engine)
55 #define LOG_NG LOG_STREAM(info, log_engine)
58 #define ERR_WML LOG_STREAM(err, log_wml)
67 , previous_free_hex_()
72 , unselected_paths_(false)
73 , unselected_reach_(false)
77 , reachmap_invalid_(false)
78 , show_partial_move_(false)
79 , teleport_selected_(false)
80 , preventing_units_highlight_(false)
99 const unsigned threshold_1080p = 14;
101 const double scale_factor = threshold_1080p / std::hypot(1080,1920);
102 return static_cast<int>(screen_diagonal * scale_factor);
120 if(mini_loc.
valid()) {
138 const double drag_distance =
139 std::pow(
static_cast<double>(
drag_from_.x - pos.x), 2) +
140 std::pow(
static_cast<double>(
drag_from_.y - pos.y), 2);
150 bool selected_hex_has_my_unit = found_unit.valid() && found_unit.get_shared_ptr()->side() ==
side_num_;
196 selected_hex_has_my_unit = found_unit.valid();
250 if (attack_from.
valid()) {
253 else if (!mouseover_unit &&
266 && mouseover_unit->side() ==
side_num_ ) {
291 if (attack_from.
valid()) {
321 && mouseover_unit.
valid()
348 !
gui().fogged(un->get_location()))
354 if(!
gui().viewing_team().is_enemy(un->side()) && !
pc_.
get_teams()[un->side() - 1].is_ai()) {
373 std::unique_ptr<unit_movement_resetter> move_reset;
375 move_reset = std::make_unique<unit_movement_resetter>(*un);
414 lk->mouse_over_hex_callback(new_hex);
447 bool selected_hex_has_unit;
502 if(attack_from.
valid()) {
539 if(attack_from.
valid()) {
608 static const std::array<const std::string, 6> buttons = {
617 if (
gui().view_locked() || button < SDL_BUTTON_LEFT || button > buttons.size()) {
624 lk->mouse_button_callback(
loc, buttons[button], (event.state == SDL_RELEASED ?
"up" :
"down"));
633 return lk->mouse_button_callback(
loc, buttons[button],
"click");
667 return it.
valid() ? &*it :
nullptr;
673 return it.
valid() ? &*it :
nullptr;
695 std::set<int> attackable_distances;
703 bool source_eligible = source_unit.
valid();
704 if(!source_eligible) {
710 if(!source_eligible) {
719 if(!source_eligible) {
725 source_eligible &= source_unit->attacks_left() != 0;
726 if(!source_eligible) {
737 bool target_eligible = target_unit.
valid();
738 if(!target_eligible) {
743 target_eligible &= viewer.
is_enemy(target_unit->side());
744 if(!target_eligible) {
749 assert(source_unit->side() != target_unit->side());
751 target_eligible &= !target_unit->incapacitated();
752 if(!target_eligible) {
756 for (
const auto& attack : source_unit->attacks()) {
757 for (
int i = attack.min_range();
i <= attack.max_range(); ++
i) {
758 attackable_distances.insert(
i);
764 if(attackable_distances.empty()) {
769 if(*attackable_distances.rbegin() > 1){
791 int best_rating = 100;
796 for(std::size_t
n = 0;
n < adj.size(); ++
n) {
808 unsigned int difference = std::abs(
static_cast<int>(
static_cast<int>(preferred) -
n));
809 if(difference > ndirections / 2) {
810 difference = ndirections - difference;
813 unsigned int second_difference = std::abs(
static_cast<int>(
static_cast<int>(second_preferred) -
n));
814 if(second_difference > ndirections / 2) {
815 second_difference = ndirections - second_difference;
818 const int rating = difference * 2 + (second_difference > difference);
819 if(rating < best_rating || res.
valid() ==
false) {
820 best_rating = rating;
841 un->
get_location(), go_to, 10000.0, calc, board.
map().
w(), board.
map().
h(), &allowed_teleports);
868 if(clicked_u && (!selected_u || selected_u->side() !=
side_num_ ||
869 (clicked_u->side() ==
side_num_ && clicked_u->id() != selected_u->id()))
914 if(clicked_u && (!selected_u || selected_u->side() !=
side_num_ ||
915 (clicked_u->side() ==
side_num_ && clicked_u->id() != selected_u->id()))
943 const unit* clicked_u =
nullptr;
1192 bool invisible = u->invisible(u->get_location());
1194 if(!
gui_->
fogged(u->get_location()) && !u->incapacitated() && !invisible) {
1198 if(
path.destinations.find(hex) !=
path.destinations.end()) {
1244 bool interrupted =
false;
1245 if(steps.size() > 1) {
1248 interrupted = interrupted || num_moves + 1 < steps.size();
1256 return !interrupted;
1280 interrupted =
false;
1292 if(u && u->can_recruit() && u->side() ==
gui().viewing_team().side()
1295 _(
"You cannot move your leader away from the keep with some planned recruits or recalls left."));
1300 LOG_NG <<
"move unit along route from " << steps.front() <<
" to " << steps.back();
1309 if(interrupted && moves + 1 < steps.size()) {
1311 select_hex(steps[moves],
false,
false,
false);
1345 pc_.
get_whiteboard()->save_temp_attack(attacker_loc, defender_loc, weapon_choice);
1352 for(
unsigned int i = 0;
i < attacker->attacks().
size();
i++) {
1354 if(attacker->attacks()[
i].attack_weight() > 0) {
1362 if(!bc_vector.empty() && bc.
better_attack(bc_vector[best], 0.5)) {
1364 best = bc_vector.size();
1367 bc_vector.emplace_back(std::move(bc));
1379 std::vector<battle_context> bc_vector;
1380 std::vector<gui2::widget_data> bc_widget_data_vector;
1382 int leadership_bonus = 0;
1387 attacker = board.
units().
find(attacker_loc);
1388 defender = board.
units().
find(defender_loc);
1390 if(!attacker || !defender) {
1391 if (!attacker) {
ERR_NG <<
"Attacker is missing, can't attack";}
1392 if (!defender) {
ERR_NG <<
"Defender is missing, can't attack";}
1398 if (bc_vector.empty()) {
1404 static const config empty;
1408 for(
const auto& weapon : bc_vector) {
1415 if(leadership_bonus == 0) {
1423 const std::string attw_name = !attacker_weapon.
name().
empty() ? attacker_weapon.
name() :
" ";
1424 const std::string defw_name = !defender_weapon.
name().
empty() ? defender_weapon.
name() :
" ";
1426 std::string range = attacker_weapon.
range().empty() ? defender_weapon.
range() : attacker_weapon.
range();
1427 if(!range.empty()) {
1432 attacker->get_location(), defender->get_location(),
true, defender_stats.
weapon);
1435 defender->get_location(), attacker->get_location(),
false, attacker_stats.
weapon);
1438 std::string attw_type = !(types).empty() ? types : attacker_weapon.
type();
1439 if(!attw_type.empty()) {
1443 std::string defw_type = !(def_types).empty() ? def_types : defender_weapon.
type();
1444 if(!defw_type.empty()) {
1448 const std::set<std::string> checking_tags_other = {
"damage_type",
"disable",
"berserk",
"drains",
1449 "heal_on_hit",
"plague",
"slow",
"petrifies",
"firststrike",
"poison"};
1455 bool defender_attack = !(defender_weapon.
name().
empty() && defender_weapon.
damage() == 0
1457 std::string defw_specials = defender_attack ? defender_weapon.
weapon_specials() :
"";
1458 std::string defw_specials_dmg
1460 std::string defw_specials_atk
1462 std::string defw_specials_cth
1464 std::string defw_specials_others
1467 if(!attw_specials.empty()) {
1468 attw_specials =
" " + attw_specials;
1470 if(!attw_specials_dmg.empty()) {
1471 attw_specials_dmg =
" " + attw_specials_dmg;
1473 if(!attw_specials_atk.empty()) {
1474 attw_specials_atk =
" " + attw_specials_atk;
1476 if(!attw_specials_cth.empty()) {
1477 attw_specials_cth =
" " + attw_specials_cth;
1479 if(!attw_specials_others.empty()) {
1480 attw_specials_others
1483 if(!defw_specials.empty()) {
1484 defw_specials =
" " + defw_specials;
1486 if(!defw_specials_dmg.empty()) {
1487 defw_specials_dmg =
" " + defw_specials_dmg;
1489 if(!defw_specials_atk.empty()) {
1490 defw_specials_atk =
" " + defw_specials_atk;
1492 if(!defw_specials_cth.empty()) {
1493 defw_specials_cth =
" " + defw_specials_cth;
1495 if(!defw_specials_others.empty()) {
1496 defw_specials_others
1500 std::stringstream attacker_stats_str, defender_stats_str, attacker_tooltip, defender_tooltip;
1505 << attw_type <<
"\n"
1507 << attw_specials <<
"\n"
1510 attacker_tooltip <<
_(
"Weapon: ") <<
markup::bold(attw_name) <<
"\n"
1511 <<
_(
"Type: ") << attw_type <<
"\n"
1514 <<
_(
"Chance to hit: ")
1519 << defw_type <<
"\n"
1521 << defw_specials <<
"\n"
1524 defender_tooltip <<
_(
"Weapon: ") <<
markup::bold(defw_name) <<
"\n"
1525 <<
_(
"Type: ") << defw_type <<
"\n"
1528 <<
_(
"Chance to hit: ")
1535 item[
"use_markup"] =
"true";
1537 item[
"label"] = attacker_weapon.
icon();
1538 data.emplace(
"attacker_weapon_icon", item);
1540 item[
"tooltip"] = attacker_tooltip.str();
1541 item[
"label"] = attacker_stats_str.str();
1542 data.emplace(
"attacker_weapon", item);
1543 item[
"tooltip"] =
"";
1547 data.emplace(
"range", item);
1549 item[
"tooltip"] = defender_attack ? defender_tooltip.str() :
"";
1550 item[
"label"] = defender_stats_str.str();
1551 data.emplace(
"defender_weapon", item);
1553 item[
"tooltip"] =
"";
1554 item[
"label"] = defender_weapon.
icon();
1555 data.emplace(
"defender_weapon_icon", item);
1557 bc_widget_data_vector.emplace_back(
data);
1574 }
catch(
const std::bad_alloc&) {
1575 lg::log_to_chat() <<
"Memory exhausted a unit has either a lot hitpoints or a negative amount.\n";
1576 ERR_WML <<
"Memory exhausted a unit has either a lot hitpoints or a negative amount.";
1588 unit* attacker =
nullptr;
1589 const unit* defender =
nullptr;
1590 std::vector<battle_context> bc_vector;
1594 if(!attacker_it || attacker_it->side() !=
side_num_ || attacker_it->incapacitated()) {
1599 if(!defender_it ||
current_team().is_enemy(defender_it->side()) ==
false || defender_it->incapacitated()) {
1605 attacker = &*attacker_it;
1606 defender = &*defender_it;
1609 if(std::size_t(choice) >= bc_vector.size()) {
1653 std::set<map_location> res;
1680 if(
current_team().is_enemy(
gui().viewing_team().side()) && it->invisible(it->get_location())) {
1684 if(it->get_hidden()) {
1708 if(it == units.
begin()) {
1714 if(it == units.
end()) {
int under_leadership(const unit &u, const map_location &loc, const_attack_ptr weapon, const_attack_ptr opp_weapon)
Tests if the unit at loc is currently affected by leadership.
Various functions that implement attacks and attack calculations.
Various functions related to moving units.
std::string weapon_specials() const
Returns a comma-separated string of active names for the specials of *this.
std::string weapon_specials_value(const std::set< std::string > &checking_tags) const
const std::string & range() const
const std::string & type() const
const t_string & name() const
specials_context_t specials_context(unit_const_ptr self, unit_const_ptr other, const map_location &unit_loc, const map_location &other_loc, bool attacking, const_attack_ptr other_attack) const
const std::string & icon() const
std::pair< std::string, int > effective_damage_type() const
The type of attack used and the resistance value that does the most damage.
Computes the statistics of a battle between an attacker and a defender unit.
const battle_context_unit_stats & get_attacker_stats() const
This method returns the statistics of the attacker.
bool better_attack(class battle_context &that, double harm_weight)
Given this harm_weight, is this attack better than that?
A config object defines a single node in a WML file, with access to child nodes.
can_move_result unit_can_move(const unit &u) const
Work out what u can do - this does not check which player's turn is currently active,...
const team & viewing_team() const
void invalidate_game_status()
Function to invalidate the game status displayed on the sidebar.
void scroll_to_tile(const map_location &loc, SCROLL_TYPE scroll_type=ONSCREEN, bool check_fogged=true, bool force=true)
Scroll such that location loc is on-screen.
bool fogged(const map_location &loc) const
Returns true if location (x,y) is covered in fog.
void invalidate_all()
Function to invalidate all tiles.
map_location minimap_location_on(int x, int y)
given x,y co-ordinates of the mouse, will return the location of the hex in the minimap that the mous...
rect map_area() const
Returns the area used for the map.
map_location hex_clicked_on(int x, int y) const
given x,y co-ordinates of an onscreen pixel, will return the location of the hex that this pixel corr...
bool scroll(const point &amount, bool force=false)
Scrolls the display by amount pixels.
bool minimap_scrolling_
minimap scrolling (scroll-drag) state flag
map_location last_hex_
last highlighted hex
map_location drag_from_hex_
Drag start or mouse-down map location.
bool simple_warp_
MMB click (on game map) state flag.
bool mouse_motion_default(int x, int y, bool update)
This handles minimap scrolling and click-drag.
bool dragging_started_
Actual drag flag.
bool dragging_touch_
Finger drag init flag.
point drag_from_
Drag start position.
void disable_units_highlight()
Use this to disable hovering an unit from highlighting its movement range.
int fill_weapon_choices(std::vector< battle_context > &bc_vector, const unit_map::iterator &attacker, const unit_map::iterator &defender)
bool move_unit_along_current_route()
Moves a unit along the currently cached route.
bool mouse_button_event(const SDL_MouseButtonEvent &event, uint8_t button, map_location loc, bool click=false) override
map_location previous_hex_
void show_reach_for_unit(const unit_ptr &un)
Highlight the hexes that a unit can move to.
game_display & gui() override
Due to the way this class is constructed we can assume that the display* gui_ member actually points ...
void select_or_action(bool browse)
void attack_enemy_(const map_location &attacker_loc, const map_location &defender_loc, int choice)
void save_whiteboard_attack(const map_location &attacker_loc, const map_location &defender_loc, int weapon_choice)
int show_attack_dialog(const map_location &attacker_loc, const map_location &defender_loc, const map_location &attacker_src)
void set_current_paths(const pathfind::paths &new_paths)
pathfind::marked_route get_route(const unit *un, map_location go_to, const team &team) const
unit_map::iterator selected_unit()
static mouse_handler * singleton_
void enable_units_highlight()
When unit highlighting is disabled, call this when the mouse no longer hovers any unit to enable high...
mouse_handler(play_controller &pc)
void touch_motion(int x, int y, const bool browse, bool update=false, map_location loc=map_location::null_location()) override
void attack_enemy(const map_location &attacker_loc, const map_location &defender_loc, int choice)
std::set< map_location > get_adj_enemies(const map_location &loc, int side) const
map_location previous_free_hex_
void select_hex(const map_location &hex, const bool browse, const bool highlight=true, const bool fire_event=true, const bool force_unhighlight=false)
unit_map::const_iterator find_unit(const map_location &hex) const
map_location current_unit_attacks_from(const map_location &loc) const
bool right_click_show_menu(int x, int y, const bool browse) override
Called in the default right_click when the context menu is about to be shown, can be used for preproc...
map_location selected_hex_
pathfind::marked_route current_route_
const map_location hovered_hex() const
Uses SDL and game_display::hex_clicked_on to fetch the hex the mouse is hovering, if applicable.
bool unit_in_cycle(const unit_map::const_iterator &it)
void touch_action(const map_location hex, bool browse) override
void set_side(int side_number)
void move_action(bool browse) override
Overridden in derived class.
pathfind::paths current_paths_
If non-empty, current_paths_.destinations contains a cache of highlighted hexes, likely the movement ...
bool hex_hosts_unit(const map_location &hex) const
Unit exists on the hex, no matter if friend or foe.
int drag_threshold() const override
Minimum dragging distance to fire the drag&drop.
unit * find_unit_nonowning(const map_location &hex)
bool preventing_units_highlight_
void cycle_units(const bool browse, const bool reverse=false)
std::size_t move_unit_along_route(const std::vector< map_location > &steps, bool &interrupted)
Moves a unit across the board for a player.
void mouse_motion(int x, int y, const bool browse, bool update=false, map_location loc=map_location::null_location()) override
Use update to force an update of the mouse state.
virtual const std::vector< team > & teams() const override
unit_map::iterator find_visible_unit(const map_location &loc, const team ¤t_team, bool see_all=false)
virtual const unit_map & units() const override
virtual const gamemap & map() const override
void display_unit_hex(map_location hex)
Change the unit to be displayed in the sidebar.
virtual void highlight_hex(map_location hex) override
Function to highlight a location.
void set_attack_indicator(const map_location &src, const map_location &dst)
Set the attack direction indicator.
void set_route(const pathfind::marked_route *route)
Sets the route along which footsteps are drawn to show movement of a unit.
virtual void select_hex(map_location hex) override
Function to display a location as selected.
void highlight_reach(const pathfind::paths &paths_list)
Sets the paths that are currently displayed as available for the unit to move along.
void clear_attack_indicator()
void highlight_another_reach(const pathfind::paths &paths_list, const map_location &goal=map_location::null_location())
Add more paths to highlight.
bool unhighlight_reach()
Reset highlighting of paths.
pump_result_t fire(const std::string &event, const entity_location &loc1=entity_location::null_entity, const entity_location &loc2=entity_location::null_entity, const config &data=config())
Function to fire an event.
std::unique_ptr< game_lua_kernel > lua_kernel_
int w() const
Effective map width.
int h() const
Effective map height.
bool on_board(const map_location &loc) const
Tell if a location is on the map.
bool is_keep(const map_location &loc) const
bool show(const unsigned auto_close_time=0)
Shows the window.
int get_selected_weapon() const
std::vector< team > & get_teams()
const unit_map & get_units() const
const tod_manager & get_tod_manager() const
int current_side() const
Returns the number of the side whose turn it is.
std::shared_ptr< wb::manager > get_whiteboard() const
const gamemap & get_map() const
game_events::wml_event_pump & pump()
static config get_attack(const map_location &a, const map_location &b, int att_weapon, int def_weapon, const std::string &attacker_type_id, const std::string &defender_type_id, int attacker_lvl, int defender_lvl, const std::size_t turn, const time_of_day &t)
static bool run_and_throw(const std::string &commandname, const config &data, action_spectator &spectator=get_default_spectator())
This class stores all the data for a single 'side' (in game nomenclature).
void set_action_bonus_count(const int count)
bool is_enemy(int n) const
const time_of_day & get_time_of_day(int for_turn=0) const
Returns global time of day for the passed turn.
void set_selecting()
Sets the animation state to that when the unit is selected.
Container associating units to locations.
unit_iterator find(std::size_t id)
This class represents a single unit of a specific type.
Definitions for the interface to Wesnoth Markup Language (WML).
static std::string _(const char *str)
const std::string & type_id() const
The id of this unit's type.
bool get_hidden() const
Gets whether this unit is currently hidden on the map.
int side() const
The side this unit belongs to.
unit_animation_component & anim_comp() const
const map_location & get_location() const
The current map location this unit is at.
void set_goto(const map_location &new_goto)
Sets this unit's long term destination.
symbol_table string_table
void get_adjacent_tiles(const map_location &a, map_location *res)
Function which, given a location, will place all adjacent locations in res.
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.
Standard logging facilities (interface).
static lg::log_domain log_engine("engine")
static lg::log_domain log_wml("wml")
void teleport_unit_and_record(const map_location &teleport_from, const map_location &teleport_to, move_unit_spectator *)
Teleports a unit across the board and enters the synced context.
std::size_t move_unit_and_record(const std::vector< map_location > &steps, bool continued_move, bool *interrupted)
Wrapper around the other overload.
void set(CURSOR_TYPE type)
Use the default parameter to reset cursors.
Handling of system events.
const std::string unicode_em_dash
const std::string weapon_numbers_sep
color_t red_to_green(double val, bool for_text)
Return a color corresponding to the value val red for val=0.0 to green for val=100....
bool fire_event(const ui_event event, const std::vector< std::pair< widget *, ui_event >> &event_chain, widget *dispatcher, widget *w, F &&... params)
Helper function for fire_event.
unsigned screen_width
The screen resolution and pixel pitch should be available for all widgets since their drawing method ...
std::map< std::string, widget_item > widget_data
std::map< std::string, t_string > widget_item
void show_transient_message(const std::string &title, const std::string &message, const std::string &image, const bool message_use_markup, const bool title_use_markup)
Shows a transient message to the user.
std::stringstream & log_to_chat()
Use this to show WML errors in the ingame chat.
std::string italic(Args &&... data)
Applies italic Pango markup to the input.
std::string bold(Args &&... data)
Applies bold Pango markup to the input.
std::string span_color(const color_t &color, Args &&... data)
Applies Pango markup to the input specifying its display color.
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)
marked_route mark_route(const plain_route &rt, bool update_move_cost)
Add marks on a route rt assuming that the unit located at the first hex of rt travels along it.
const teleport_map get_teleport_locations(const unit &u, const team &viewing_team, bool see_all, bool ignore_units, bool check_vision)
uint32_t get_mouse_state(int *x, int *y)
A wrapper for SDL_GetMouseState that gives coordinates in draw space.
point get_mouse_location()
Returns the current mouse location in draw space.
void play_UI_sound(const std::string &files)
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
bool contains(const Container &container, const Value &value)
Returns true iff value is found in container.
std::shared_ptr< bool > whiteboard_lock
std::shared_ptr< const attack_type > const_attack_ptr
std::shared_ptr< unit > unit_ptr
Define the game's event mechanism.
This file contains the settings handling of the widget library.
rect dst
Location on the final composed sheet.
rect src
Non-transparent portion of the surface to compose.
Structure describing the statistics of a unit involved in the battle.
unsigned int num_blows
Effective number of blows, takes swarm into account.
const_attack_ptr weapon
The weapon used by the unit to attack the opponent, or nullptr if there is none.
int damage
Effective damage of the weapon (all factors accounted for).
bool disable
Attack has disable special.
unsigned int chance_to_hit
Effective chance to hit as a percentage (all factors accounted for).
int attack_num
Index into unit->attacks() or -1 for none.
The basic class for representing 8-bit RGB or RGBA colour values.
Encapsulates the map of the game.
direction
Valid directions which can be moved in our hexagonal world.
direction get_relative_dir(const map_location &loc, map_location::RELATIVE_DIR_MODE mode) const
static const map_location & null_location()
Structure which holds a single route and marks for special events.
std::vector< map_location > & steps
void insert(const map_location &)
bool contains(const map_location &) const
const_iterator find(const map_location &) const
Object which contains all the possible locations a unit can move to, with associated best routes to t...
Structure which holds a single route between one location and another.
bool contains(int x, int y) const
Whether the given point lies within the rectangle.
This object is used to temporary move a unit in the unit map, swapping out any unit that is already t...
pointer get_shared_ptr() const
This is exactly the same as operator-> but it's slightly more readable, and can replace &*iter syntax...
ONLY IF whiteboard is currently active, applies the planned unit map for the duration of the struct's...
Ensures that the real unit map is active for the duration of the struct's life.
static map_location::direction n
Contains typedefs for the whiteboard.