diff --git a/doc/MAGIC.md b/doc/MAGIC.md index 2e693724c9259..e7486a851b94c 100644 --- a/doc/MAGIC.md +++ b/doc/MAGIC.md @@ -956,6 +956,7 @@ Character status value | Description `ARMOR_HEAT` | `ARMOR_STAB` | `REGEN_HP` | Affects the rate the monster recovers hp. +`VISION_RANGE` | Affects monster vision range, both day and night one. `SPEED` | Affects the base speed of the monster. `LUMINATION` | Affects monster luminance diff --git a/doc/NPCs.md b/doc/NPCs.md index 5300a3454f3e7..409d35221216b 100644 --- a/doc/NPCs.md +++ b/doc/NPCs.md @@ -1383,7 +1383,7 @@ _some functions support array arguments or kwargs, denoted with square brackets | npc_value() | ✅ | ✅ | u, n | Return NPC value toward opposite talker.

Example:
`{ "math": [ "n_npc_value()", "+=", "2" ] }`| | vitamin(`s`/`v`) | ✅ | ✅ | u, n | Return or set the characters vitamin level.
Argument is vitamin ID.

Example:
`{ "math": [ "u_vitamin('mutagen')", "=", "0" ] }`| | warmth(`s`/`v`) | ✅ | ❌ | u, n | Return the characters warmth on a body part.
Argument is bodypart ID.

Example:
The value displayed in-game is calculated as follows.
`"{ "math": [ "u_warmth_in_game", "=", "(u_warmth('torso') / 100) * 2 - 100"] }`| -| vision_range() | ✅ | ❌ | u, n | Return the character's visual range, adjusted by their mutations, effects, and other issues.

Example:
`"{ "math": [ "u_vision_range", "<", "30"] }`| +| vision_range() | ✅ | ❌ | u, n | Return the character's or monsters visual range, adjusted by their mutations, effects, and other issues.

Example:
`"{ "math": [ "n_vision_range()", "<", "30"] }`| | weather(`s`) | ✅ | ✅ | N/A
(global) | Return or set a weather aspect

Aspect must be one of:
`temperature` (in Kelvin),
`humidity` (as percentage),
`pressure` (in millibar),
`windpower` (in mph).
`precipitation` (in mm / h) either 0.5 (very_light ), 1.5 (light), or 3 (heavy). Read only.

Temperature conversion functions are available: `celsius()`, `fahrenheit()`, `from_celsius()`, and `from_fahrenheit()`.

Examples:
`{ "math": [ "weather('temperature')", "<", "from_fahrenheit( 33 )" ] }`
`{ "math": [ "fahrenheit( weather('temperature') )", "==", "21" ] }`| | damage_level() | ✅ | ❌ | u, n | Return the damage level of the talker, which must be an item.

Example:
`"condition": { "math": [ "n_damage_level()", "<", "1" ] }`| | climate_control_str_heat() | ✅ | ❌ | u, n | return amount of heat climate control that character currently has (character feels better in warm places with it), in warmth points; default 0, affected by CLIMATE_CONTROL_HEAT enchantment.

Example:
`"condition": { "math": [ "u_climate_control_str_heat()", "<", "0" ] }`| diff --git a/src/magic_enchantment.cpp b/src/magic_enchantment.cpp index dd9c0d90f63b6..5baf8e186c28f 100644 --- a/src/magic_enchantment.cpp +++ b/src/magic_enchantment.cpp @@ -95,6 +95,7 @@ namespace io case enchant_vals::mod::ATTACK_NOISE: return "ATTACK_NOISE"; case enchant_vals::mod::SHOUT_NOISE: return "SHOUT_NOISE"; case enchant_vals::mod::FOOTSTEP_NOISE: return "FOOTSTEP_NOISE"; + case enchant_vals::mod::VISION_RANGE: return "VISION_RANGE"; case enchant_vals::mod::SIGHT_RANGE_ELECTRIC: return "SIGHT_RANGE_ELECTRIC"; case enchant_vals::mod::MOTION_VISION_RANGE: return "MOTION_VISION_RANGE"; case enchant_vals::mod::SIGHT_RANGE_FAE: return "SIGHT_RANGE_FAE"; @@ -324,6 +325,7 @@ bool enchantment::is_monster_relevant() const pair_values.first == enchant_vals::mod::ARMOR_HEAT || pair_values.first == enchant_vals::mod::ARMOR_STAB || pair_values.first == enchant_vals::mod::REGEN_HP || + pair_values.first == enchant_vals::mod::VISION_RANGE || pair_values.first == enchant_vals::mod::SPEED || pair_values.first == enchant_vals::mod::LUMINATION ) { return true; @@ -344,6 +346,7 @@ bool enchantment::is_monster_relevant() const pair_values.first == enchant_vals::mod::ARMOR_HEAT || pair_values.first == enchant_vals::mod::ARMOR_STAB || pair_values.first == enchant_vals::mod::REGEN_HP || + pair_values.first == enchant_vals::mod::VISION_RANGE || pair_values.first == enchant_vals::mod::SPEED || pair_values.first == enchant_vals::mod::LUMINATION ) { return true; diff --git a/src/magic_enchantment.h b/src/magic_enchantment.h index 40bbb2aab53b8..bf6bd747e3dda 100644 --- a/src/magic_enchantment.h +++ b/src/magic_enchantment.h @@ -71,6 +71,7 @@ enum class mod : int { ATTACK_NOISE, SHOUT_NOISE, FOOTSTEP_NOISE, + VISION_RANGE, SIGHT_RANGE_ELECTRIC, MOTION_VISION_RANGE, SIGHT_RANGE_FAE, diff --git a/src/math_parser_diag.cpp b/src/math_parser_diag.cpp index bbf429c58058c..24e85a6b734e7 100644 --- a/src/math_parser_diag.cpp +++ b/src/math_parser_diag.cpp @@ -1478,6 +1478,10 @@ std::function vision_range_eval( char scope, talker const *const actor = d.actor( beta ); if( Character const *const chr = actor->get_character(); chr != nullptr ) { return chr->unimpaired_range(); + } else if( monster const *const mon = actor->get_monster(); mon != nullptr ) { + map &here = get_map(); + tripoint_bub_ms tripoint = get_map().bub_from_abs( mon->get_location() ); + return mon->sight_range( here.ambient_light_at( tripoint ) ); } debugmsg( "Tried to access vision range of a non-Character talker" ); return 0; diff --git a/src/monster.cpp b/src/monster.cpp index 1ae2e6f78e4b7..8ade126638dbb 100644 --- a/src/monster.cpp +++ b/src/monster.cpp @@ -1385,13 +1385,15 @@ int monster::sight_range( const float light_level ) const } static const float default_daylight = default_daylight_level(); if( light_level == 0 ) { - return type->vision_night; + return calculate_by_enchantment( type->vision_night, enchant_vals::mod::VISION_RANGE, true ); } else if( light_level >= default_daylight ) { - return type->vision_day; + return calculate_by_enchantment( type->vision_day, enchant_vals::mod::VISION_RANGE, true ); } int range = ( light_level * type->vision_day + ( default_daylight - light_level ) * type->vision_night ) / default_daylight; + range = calculate_by_enchantment( range, enchant_vals::mod::VISION_RANGE, true ); + return range; }