Skip to content
Permalink
Branch: master
Find file Copy path
Find file Copy path
9331 lines (8292 sloc) 358 KB
#include "iuse.h"
#include <climits>
#include <algorithm>
#include <cmath>
#include <cstdlib>
#include <set>
#include <sstream>
#include <vector>
#include <array>
#include <exception>
#include <functional>
#include <iterator>
#include <list>
#include <map>
#include <utility>
#include <unordered_map>
#include <unordered_set>
#include "action.h"
#include "artifact.h"
#include "avatar.h"
#include "calendar.h"
#include "cata_utility.h"
#include "coordinate_conversions.h"
#include "debug.h"
#include "effect.h" // for weed_msg
#include "explosion.h"
#include "event.h"
#include "field.h"
#include "fungal_effects.h"
#include "game.h"
#include "game_inventory.h"
#include "iexamine.h"
#include "inventory.h"
#include "iteminfo_query.h"
#include "iuse_actor.h" // For firestarter
#include "json.h"
#include "line.h"
#include "map.h"
#include "map_iterator.h"
#include "mapdata.h"
#include "martialarts.h"
#include "messages.h"
#include "monattack.h"
#include "mongroup.h"
#include "morale_types.h"
#include "mtype.h"
#include "mutation.h"
#include "npc.h"
#include "output.h"
#include "overmap.h"
#include "overmapbuffer.h"
#include "player.h"
#include "recipe_dictionary.h"
#include "requirements.h"
#include "rng.h"
#include "sounds.h"
#include "speech.h"
#include "string_formatter.h"
#include "string_input_popup.h"
#include "text_snippets.h"
#include "translations.h"
#include "trap.h"
#include "ui.h"
#include "vehicle.h"
#include "vpart_position.h"
#include "vpart_range.h"
#include "veh_type.h"
#include "weather.h"
#include "bodypart.h"
#include "character.h"
#include "color.h"
#include "creature.h"
#include "damage.h"
#include "enums.h"
#include "game_constants.h"
#include "int_id.h"
#include "inventory_ui.h"
#include "item.h"
#include "item_location.h"
#include "itype.h"
#include "monster.h"
#include "optional.h"
#include "pimpl.h"
#include "player_activity.h"
#include "pldata.h"
#include "recipe.h"
#include "ret_val.h"
#include "stomach.h"
#include "string_id.h"
#include "weather_gen.h"
#include "type_id.h"
#include "options.h"
#include "flat_set.h"
#include "handle_liquid.h"
#include "item_group.h"
#include "omdata.h"
#include "point.h"
#define RADIO_PER_TURN 25 // how many characters per turn of radio
#include "iuse_software.h"
const mtype_id mon_bee( "mon_bee" );
const mtype_id mon_blob( "mon_blob" );
const mtype_id mon_cat( "mon_cat" );
const mtype_id mon_hologram( "mon_hologram" );
const mtype_id mon_dog( "mon_dog" );
const mtype_id mon_dog_thing( "mon_dog_thing" );
const mtype_id mon_fly( "mon_fly" );
const mtype_id mon_hallu_multicooker( "mon_hallu_multicooker" );
const mtype_id mon_shadow( "mon_shadow" );
const mtype_id mon_spore( "mon_spore" );
const mtype_id mon_vortex( "mon_vortex" );
const mtype_id mon_wasp( "mon_wasp" );
const mtype_id mon_cow( "mon_cow" );
const skill_id skill_firstaid( "firstaid" );
const skill_id skill_tailor( "tailor" );
const skill_id skill_survival( "survival" );
const skill_id skill_cooking( "cooking" );
const skill_id skill_mechanics( "mechanics" );
const skill_id skill_archery( "archery" );
const skill_id skill_computer( "computer" );
const skill_id skill_cutting( "cutting" );
const skill_id skill_fabrication( "fabrication" );
const skill_id skill_electronics( "electronics" );
const skill_id skill_melee( "melee" );
const species_id ROBOT( "ROBOT" );
const species_id HALLUCINATION( "HALLUCINATION" );
const species_id ZOMBIE( "ZOMBIE" );
const species_id FUNGUS( "FUNGUS" );
const species_id INSECT( "INSECT" );
const efftype_id effect_adrenaline( "adrenaline" );
const efftype_id effect_antibiotic( "antibiotic" );
const efftype_id effect_antibiotic_visible( "antibiotic_visible" );
const efftype_id effect_asthma( "asthma" );
const efftype_id effect_attention( "attention" );
const efftype_id effect_beartrap( "beartrap" );
const efftype_id effect_bite( "bite" );
const efftype_id effect_bleed( "bleed" );
const efftype_id effect_blind( "blind" );
const efftype_id effect_bloodworms( "bloodworms" );
const efftype_id effect_boomered( "boomered" );
const efftype_id effect_bouldering( "bouldering" );
const efftype_id effect_brainworms( "brainworms" );
const efftype_id effect_cig( "cig" );
const efftype_id effect_contacts( "contacts" );
const efftype_id effect_corroding( "corroding" );
const efftype_id effect_crushed( "crushed" );
const efftype_id effect_cureall( "cureall" );
const efftype_id effect_datura( "datura" );
const efftype_id effect_dazed( "dazed" );
const efftype_id effect_dermatik( "dermatik" );
const efftype_id effect_docile( "docile" );
const efftype_id effect_downed( "downed" );
const efftype_id effect_drunk( "drunk" );
const efftype_id effect_earphones( "earphones" );
const efftype_id effect_flushot( "flushot" );
const efftype_id effect_foodpoison( "foodpoison" );
const efftype_id effect_formication( "formication" );
const efftype_id effect_fungus( "fungus" );
const efftype_id effect_glowing( "glowing" );
const efftype_id effect_glowing_led( "glowy_led" );
const efftype_id effect_hallu( "hallu" );
const efftype_id effect_happy( "happy" );
const efftype_id effect_harnessed( "harnessed" );
const efftype_id effect_has_bag( "has_bag" );
const efftype_id effect_haslight( "haslight" );
const efftype_id effect_high( "high" );
const efftype_id effect_in_pit( "in_pit" );
const efftype_id effect_infected( "infected" );
const efftype_id effect_jetinjector( "jetinjector" );
const efftype_id effect_lack_sleep( "lack_sleep" );
const efftype_id effect_laserlocked( "laserlocked" );
const efftype_id effect_lying_down( "lying_down" );
const efftype_id effect_meth( "meth" );
const efftype_id effect_monster_armor( "monster_armor" );
const efftype_id effect_music( "music" );
const efftype_id effect_onfire( "onfire" );
const efftype_id effect_paincysts( "paincysts" );
const efftype_id effect_panacea( "panacea" );
const efftype_id effect_pet( "pet" );
const efftype_id effect_poison( "poison" );
const efftype_id effect_recover( "recover" );
const efftype_id effect_ridden( "ridden" );
const efftype_id effect_riding( "riding" );
const efftype_id effect_run( "run" );
const efftype_id effect_sad( "sad" );
const efftype_id effect_saddled( "monster_saddled" );
const efftype_id effect_sap( "sap" );
const efftype_id effect_shakes( "shakes" );
const efftype_id effect_sleep( "sleep" );
const efftype_id effect_slimed( "slimed" );
const efftype_id effect_smoke( "smoke" );
const efftype_id effect_spores( "spores" );
const efftype_id effect_stimpack( "stimpack" );
const efftype_id effect_strong_antibiotic( "strong_antibiotic" );
const efftype_id effect_strong_antibiotic_visible( "strong_antibiotic_visible" );
const efftype_id effect_stunned( "stunned" );
const efftype_id effect_teargas( "teargas" );
const efftype_id effect_tapeworm( "tapeworm" );
const efftype_id effect_teleglow( "teleglow" );
const efftype_id effect_tetanus( "tetanus" );
const efftype_id effect_tied( "tied" );
const efftype_id effect_took_anticonvulsant_visible( "took_anticonvulsant_visible" );
const efftype_id effect_took_flumed( "took_flumed" );
const efftype_id effect_took_prozac( "took_prozac" );
const efftype_id effect_took_prozac_bad( "took_prozac_bad" );
const efftype_id effect_took_prozac_visible( "took_prozac_visible" );
const efftype_id effect_took_xanax( "took_xanax" );
const efftype_id effect_took_xanax_visible( "took_xanax_visible" );
const efftype_id effect_valium( "valium" );
const efftype_id effect_visuals( "visuals" );
const efftype_id effect_weak_antibiotic( "weak_antibiotic" );
const efftype_id effect_weak_antibiotic_visible( "weak_antibiotic_visible" );
const efftype_id effect_webbed( "webbed" );
const efftype_id effect_weed_high( "weed_high" );
const efftype_id effect_magnesium_supplements( "magnesium" );
static const trait_id trait_ACIDBLOOD( "ACIDBLOOD" );
static const trait_id trait_ACIDPROOF( "ACIDPROOF" );
static const trait_id trait_ALCMET( "ALCMET" );
static const trait_id trait_CENOBITE( "CENOBITE" );
static const trait_id trait_CHLOROMORPH( "CHLOROMORPH" );
static const trait_id trait_EATDEAD( "EATDEAD" );
static const trait_id trait_EATPOISON( "EATPOISON" );
static const trait_id trait_GILLS( "GILLS" );
static const trait_id trait_HYPEROPIC( "HYPEROPIC" );
static const trait_id trait_ILLITERATE( "ILLITERATE" );
static const trait_id trait_LIGHTWEIGHT( "LIGHTWEIGHT" );
static const trait_id trait_MARLOSS_AVOID( "MARLOSS_AVOID" );
static const trait_id trait_MARLOSS_BLUE( "MARLOSS_BLUE" );
static const trait_id trait_MARLOSS( "MARLOSS" );
static const trait_id trait_MARLOSS_YELLOW( "MARLOSS_YELLOW" );
static const trait_id trait_MASOCHIST( "MASOCHIST" );
static const trait_id trait_MASOCHIST_MED( "MASOCHIST_MED" );
static const trait_id trait_M_DEPENDENT( "M_DEPENDENT" );
static const trait_id trait_MYOPIC( "MYOPIC" );
static const trait_id trait_NOPAIN( "NOPAIN" );
static const trait_id trait_PSYCHOPATH( "PSYCHOPATH" );
static const trait_id trait_SPIRITUAL( "SPIRITUAL" );
static const trait_id trait_THRESH_MARLOSS( "THRESH_MARLOSS" );
static const trait_id trait_THRESH_MYCUS( "THRESH_MYCUS" );
static const trait_id trait_THRESH_PLANT( "THRESH_PLANT" );
static const trait_id trait_TOLERANCE( "TOLERANCE" );
static const trait_id trait_URSINE_EYE( "URSINE_EYE" );
static const quality_id AXE( "AXE" );
static const quality_id DIG( "DIG" );
struct extended_photo_def;
struct object_names_collection;
static void item_save_monsters( player &p, item &it, const std::vector<monster *> &monster_vec,
const int photo_quality );
static bool show_photo_selection( player &p, item &it, const std::string &var_name );
static bool item_read_extended_photos( item &, std::vector<extended_photo_def> &,
const std::string &,
bool = false );
static void item_write_extended_photos( item &, const std::vector<extended_photo_def> &,
const std::string & );
static std::string format_object_pair( const std::pair<std::string, int> &pair,
const std::string &article );
static std::string format_object_pair_article( const std::pair<std::string, int> &pair );
static std::string format_object_pair_no_article( const std::pair<std::string, int> &pair );
static std::string colorized_field_description_at( const tripoint &point );
static std::string colorized_trap_name_at( const tripoint &point );
static std::string colorized_ter_name_flags_at( const tripoint &point,
const std::vector<std::string> &flags = {}, const std::vector<ter_str_id> &ter_whitelist = {} );
static std::string colorized_feature_description_at( const tripoint &center_point, bool &item_found,
const units::volume min_visible_volume );
static std::string colorized_item_name( const item &item );
static std::string colorized_item_description( const item &item );
static const item get_top_item_at_point( const tripoint &point,
const units::volume min_visible_volume );
static std::string effects_description_for_creature( Creature *const creature, std::string &pose,
const std::string &pronoun_sex );
static object_names_collection enumerate_objects_around_point( const tripoint &point,
const int radius, const tripoint &bounds_center_point, const int bounds_radius,
const tripoint &camera_pos, const units::volume min_visible_volume, bool create_figure_desc,
std::unordered_set<tripoint> &ignored_points,
std::unordered_set<const vehicle *> &vehicles_recorded );
static extended_photo_def photo_def_for_camera_point( const tripoint &aim_point,
const tripoint &camera_pos,
std::vector<monster *> &monster_vec, std::vector<player *> &player_vec );
static const std::vector<std::string> camera_ter_whitelist_flags = {
"HIDE_PLACE", "FUNGUS", "TREE", "PERMEABLE", "SHRUB",
"PLACE_ITEM", "GROWTH_HARVEST", "GROWTH_MATURE", "GOES_UP",
"GOES_DOWN", "RAMP", "SHARP", "SIGN", "CLIMBABLE"
};
static const std::vector<ter_str_id> camera_ter_whitelist_types = {
ter_str_id( "t_pit_covered" ), ter_str_id( "t_grave_new" ), ter_str_id( "t_grave" ), ter_str_id( "t_pit" ),
ter_str_id( "t_pit_shallow" ), ter_str_id( "t_pit_corpsed" ), ter_str_id( "t_pit_spiked" ),
ter_str_id( "t_pit_spiked_covered" ), ter_str_id( "t_pit_glass" ), ter_str_id( "t_pit_glass" ), ter_str_id( "t_utility_light" )
};
void remove_radio_mod( item &it, player &p )
{
if( !it.has_flag( "RADIO_MOD" ) ) {
return;
}
p.add_msg_if_player( _( "You remove the radio modification from your %s!" ), it.tname() );
item mod( "radio_mod" );
p.i_add_or_drop( mod, 1 );
it.item_tags.erase( "RADIO_ACTIVATION" );
it.item_tags.erase( "RADIO_MOD" );
it.item_tags.erase( "RADIOSIGNAL_1" );
it.item_tags.erase( "RADIOSIGNAL_2" );
it.item_tags.erase( "RADIOSIGNAL_3" );
it.item_tags.erase( "RADIOCARITEM" );
}
// Checks that the player does not have an active item with LITCIG flag.
static bool check_litcig( player &u )
{
auto cigs = u.items_with( []( const item & it ) {
return it.active && it.has_flag( "LITCIG" );
} );
if( cigs.empty() ) {
return true;
}
u.add_msg_if_player( m_info, _( "You're already smoking a %s!" ), cigs[0]->tname() );
return false;
}
/* iuse methods return the number of charges expended, which is usually it->charges_to_use().
* Some items that don't normally use charges return 1 to indicate they're used up.
* Regardless, returning 0 indicates the item has not been used up,
* though it may have been successfully activated.
*/
int iuse::sewage( player *p, item *it, bool, const tripoint & )
{
if( !p->query_yn( _( "Are you sure you want to drink... this?" ) ) ) {
return 0;
}
p->add_memorial_log( pgettext( "memorial_male", "Ate a sewage sample." ),
pgettext( "memorial_female", "Ate a sewage sample." ) );
p->vomit();
if( one_in( 4 ) ) {
p->mutate();
}
return it->type->charges_to_use();
}
int iuse::honeycomb( player *p, item *it, bool, const tripoint & )
{
g->m.spawn_item( p->pos(), "wax", 2 );
return it->type->charges_to_use();
}
int iuse::royal_jelly( player *p, item *it, bool, const tripoint & )
{
p->add_effect( effect_cureall, 1_turns );
return it->type->charges_to_use();
}
int iuse::xanax( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You take some %s." ), it->tname() );
p->add_effect( effect_took_xanax, 90_minutes );
p->add_effect( effect_took_xanax_visible, rng( 70_minutes, 110_minutes ) );
return it->type->charges_to_use();
}
int iuse::caff( player *p, item *it, bool, const tripoint & )
{
p->mod_fatigue( -( it->get_comestible() ? it->get_comestible()->stim : 0 ) * 3 );
return it->type->charges_to_use();
}
int iuse::atomic_caff( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( m_good, _( "Wow! This %s has a kick." ), it->tname() );
p->mod_fatigue( -( it->get_comestible() ? it->get_comestible()->stim : 0 ) * 12 );
p->irradiate( 8, true );
return it->type->charges_to_use();
}
static constexpr time_duration alc_strength( const int strength, const time_duration &weak,
const time_duration &medium, const time_duration &strong )
{
return strength == 0 ? weak : strength == 1 ? medium : strong;
}
static int alcohol( player &p, const item &it, const int strength )
{
// Weaker characters are cheap drunks
/** @EFFECT_STR_MAX reduces drunkenness duration */
time_duration duration = alc_strength( strength, 34_minutes, 68_minutes,
90_minutes ) - ( alc_strength( strength, 36_seconds, 1_minutes, 72_seconds ) * p.str_max );
if( p.has_trait( trait_ALCMET ) ) {
duration = alc_strength( strength, 9_minutes, 18_minutes, 25_minutes ) - ( alc_strength( strength,
36_seconds, 1_minutes, 1_minutes ) * p.str_max );
// Metabolizing the booze improves the nutritional value;
// might not be healthy, and still causes Thirst problems, though
p.stomach.mod_nutr( -( abs( it.get_comestible() ? it.type->comestible->stim : 0 ) ) );
// Metabolizing it cancels out the depressant
p.stim += abs( it.get_comestible() ? it.get_comestible()->stim : 0 );
} else if( p.has_trait( trait_TOLERANCE ) ) {
duration -= alc_strength( strength, 12_minutes, 30_minutes, 45_minutes );
} else if( p.has_trait( trait_LIGHTWEIGHT ) ) {
duration += alc_strength( strength, 12_minutes, 30_minutes, 45_minutes );
}
if( !( p.has_trait( trait_ALCMET ) ) ) {
p.mod_painkiller( to_turns<int>( alc_strength( strength, 24_seconds, 48_seconds, 72_seconds ) ) );
}
p.add_effect( effect_drunk, duration );
return it.type->charges_to_use();
}
int iuse::alcohol_weak( player *p, item *it, bool, const tripoint & )
{
return alcohol( *p, *it, 0 );
}
int iuse::alcohol_medium( player *p, item *it, bool, const tripoint & )
{
return alcohol( *p, *it, 1 );
}
int iuse::alcohol_strong( player *p, item *it, bool, const tripoint & )
{
return alcohol( *p, *it, 2 );
}
/**
* Entry point for intentional bodily intake of smoke via paper wrapped one
* time use items: cigars, cigarettes, etc.
*
* @param p Player doing the smoking
* @param it the item to be smoked.
* @return Charges used in item smoked
*/
int iuse::smoking( player *p, item *it, bool, const tripoint & )
{
bool hasFire = ( p->has_charges( "fire", 1 ) );
// make sure we're not already smoking something
if( !check_litcig( *p ) ) {
return 0;
}
if( !hasFire ) {
p->add_msg_if_player( m_info, _( "You don't have anything to light it with!" ) );
return 0;
}
item cig;
if( it->typeId() == "cig" ) {
cig = item( "cig_lit", calendar::turn );
cig.item_counter = to_turns<int>( 4_minutes );
p->mod_hunger( -3 );
p->mod_thirst( 2 );
} else if( it->typeId() == "handrolled_cig" ) {
// This transforms the hand-rolled into a normal cig, which isn't exactly
// what I want, but leaving it for now.
cig = item( "cig_lit", calendar::turn );
cig.item_counter = to_turns<int>( 4_minutes );
p->mod_thirst( 2 );
p->mod_hunger( -3 );
} else if( it->typeId() == "cigar" ) {
cig = item( "cigar_lit", calendar::turn );
cig.item_counter = to_turns<int>( 12_minutes );
p->mod_thirst( 3 );
p->mod_hunger( -4 );
} else if( it->typeId() == "joint" ) {
cig = item( "joint_lit", calendar::turn );
cig.item_counter = to_turns<int>( 4_minutes );
p->mod_hunger( 4 );
p->mod_thirst( 6 );
if( p->get_painkiller() < 5 ) {
p->set_painkiller( ( p->get_painkiller() + 3 ) * 2 );
}
} else {
p->add_msg_if_player( m_bad,
_( "Please let the devs know you should be able to smoke a %s but the smoking code does not know how." ),
it->tname() );
return 0;
}
// If we're here, we better have a cig to light.
p->use_charges_if_avail( "fire", 1 );
cig.active = true;
p->inv.add_item( cig, false, true );
p->add_msg_if_player( m_neutral, _( "You light a %s." ), cig.tname() );
// Parting messages
if( it->typeId() == "joint" ) {
// Would group with the joint, but awkward to mutter before lighting up.
if( one_in( 5 ) ) {
weed_msg( *p );
}
}
if( p->get_effect_dur( effect_cig ) > 10_minutes * ( p->addiction_level( ADD_CIG ) + 1 ) ) {
p->add_msg_if_player( m_bad, _( "Ugh, too much smoke... you feel nasty." ) );
}
return it->type->charges_to_use();
}
int iuse::ecig( player *p, item *it, bool, const tripoint & )
{
if( it->typeId() == "ecig" ) {
p->add_msg_if_player( m_neutral, _( "You take a puff from your electronic cigarette." ) );
} else if( it->typeId() == "advanced_ecig" ) {
if( p->has_charges( "nicotine_liquid", 1 ) ) {
p->add_msg_if_player( m_neutral,
_( "You inhale some vapor from your advanced electronic cigarette." ) );
p->use_charges( "nicotine_liquid", 1 );
item dummy_ecig = item( "ecig", calendar::turn );
p->consume_effects( dummy_ecig );
} else {
p->add_msg_if_player( m_info, _( "You don't have any nicotine liquid!" ) );
return 0;
}
}
p->mod_thirst( 1 );
p->mod_hunger( -1 );
p->add_effect( effect_cig, 10_minutes );
if( p->get_effect_dur( effect_cig ) > 10_minutes * ( p->addiction_level( ADD_CIG ) + 1 ) ) {
p->add_msg_if_player( m_bad, _( "Ugh, too much nicotine... you feel nasty." ) );
}
return it->type->charges_to_use();
}
int iuse::antibiotic( player *p, item *it, bool, const tripoint & )
{
p->add_msg_player_or_npc( m_neutral,
_( "You take some antibiotics." ),
_( "<npcname> takes some antibiotics." ) );
if( p->has_effect( effect_tetanus ) ) {
if( one_in( 3 ) ) {
p->remove_effect( effect_tetanus );
p->add_msg_if_player( m_good, _( "The muscle spasms start to go away." ) );
} else {
p->add_msg_if_player( m_warning, _( "The medication does nothing to help the spasms." ) );
}
}
if( p->has_effect( effect_infected ) && !p->has_effect( effect_antibiotic ) ) {
p->add_msg_if_player( m_good,
_( "Maybe just placebo effect, but you feel a little better as the dose settles in." ) );
}
p->add_effect( effect_antibiotic, 12_hours );
p->add_effect( effect_antibiotic_visible, rng( 9_hours, 15_hours ) );
return it->type->charges_to_use();
}
int iuse::eyedrops( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( it->charges < it->type->charges_to_use() ) {
p->add_msg_if_player( _( "You're out of %s." ), it->tname() );
return 0;
}
p->add_msg_if_player( _( "You use your %s." ), it->tname() );
p->moves -= to_moves<int>( 10_seconds );
if( p->has_effect( effect_boomered ) ) {
p->remove_effect( effect_boomered );
p->add_msg_if_player( m_good, _( "You wash the slime from your eyes." ) );
}
return it->type->charges_to_use();
}
int iuse::fungicide( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
const bool has_fungus = p->has_effect( effect_fungus );
const bool has_spores = p->has_effect( effect_spores );
if( p->is_npc() && !has_fungus && !has_spores ) {
return 0;
}
p->add_msg_player_or_npc( _( "You use your fungicide." ), _( "<npcname> uses some fungicide" ) );
if( has_fungus && ( one_in( 3 ) ) ) {
p->remove_effect( effect_fungus );
p->add_msg_if_player( m_warning,
_( "You feel a burning sensation under your skin that quickly fades away." ) );
}
if( has_spores && ( one_in( 2 ) ) ) {
if( !p->has_effect( effect_fungus ) ) {
p->add_msg_if_player( m_warning, _( "Your skin grows warm for a moment." ) );
}
p->remove_effect( effect_spores );
int spore_count = rng( 1, 6 );
for( const tripoint &dest : g->m.points_in_radius( p->pos(), 1 ) ) {
if( spore_count == 0 ) {
break;
}
if( dest == p->pos() ) {
continue;
}
if( g->m.passable( dest ) && x_in_y( spore_count, 8 ) ) {
if( monster *const mon_ptr = g->critter_at<monster>( dest ) ) {
monster &critter = *mon_ptr;
if( g->u.sees( dest ) &&
!critter.type->in_species( FUNGUS ) ) {
add_msg( m_warning, _( "The %s is covered in tiny spores!" ),
critter.name() );
}
if( !critter.make_fungus() ) {
critter.die( p ); // counts as kill by player
}
} else {
g->summon_mon( mon_spore, dest );
}
spore_count--;
}
}
}
return it->type->charges_to_use();
}
int iuse::antifungal( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
p->add_msg_if_player( _( "You take some antifungal medication." ) );
if( p->has_effect( effect_fungus ) ) {
p->remove_effect( effect_fungus );
p->add_msg_if_player( m_warning,
_( "You feel a burning sensation under your skin that quickly fades away." ) );
}
if( p->has_effect( effect_spores ) ) {
if( !p->has_effect( effect_fungus ) ) {
p->add_msg_if_player( m_warning, _( "Your skin grows warm for a moment." ) );
}
}
return it->type->charges_to_use();
}
int iuse::antiparasitic( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
p->add_msg_if_player( _( "You take some antiparasitic medication." ) );
if( p->has_effect( effect_dermatik ) ) {
p->remove_effect( effect_dermatik );
p->add_msg_if_player( m_good, _( "The itching sensation under your skin fades away." ) );
}
if( p->has_effect( effect_tapeworm ) ) {
p->remove_effect( effect_tapeworm );
p->guts.mod_nutr( -1 ); // You just digested the tapeworm.
if( p->has_trait( trait_NOPAIN ) ) {
p->add_msg_if_player( m_good, _( "Your bowels clench as something inside them dies." ) );
} else {
p->add_msg_if_player( m_mixed, _( "Your bowels spasm painfully as something inside them dies." ) );
p->mod_pain( rng( 8, 24 ) );
}
}
if( p->has_effect( effect_bloodworms ) ) {
p->remove_effect( effect_bloodworms );
p->add_msg_if_player( _( "Your skin prickles and your veins itch for a few moments." ) );
}
if( p->has_effect( effect_brainworms ) ) {
p->remove_effect( effect_brainworms );
if( p->has_trait( trait_NOPAIN ) ) {
p->add_msg_if_player( m_good, _( "The pressure inside your head feels better already." ) );
} else {
p->add_msg_if_player( m_mixed,
_( "Your head pounds like a sore tooth as something inside of it dies." ) );
p->mod_pain( rng( 8, 24 ) );
}
}
if( p->has_effect( effect_paincysts ) ) {
p->remove_effect( effect_paincysts );
if( p->has_trait( trait_NOPAIN ) ) {
p->add_msg_if_player( m_good, _( "The stiffness in your joints goes away." ) );
} else {
p->add_msg_if_player( m_good, _( "The pain in your joints goes away." ) );
}
}
return it->type->charges_to_use();
}
int iuse::anticonvulsant( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You take some anticonvulsant medication." ) );
/** @EFFECT_STR reduces duration of anticonvulsant medication */
time_duration duration = 8_hours - p->str_cur * rng( 0_turns, 10_minutes );
if( p->has_trait( trait_TOLERANCE ) ) {
duration -= 1_hours;
}
if( p->has_trait( trait_LIGHTWEIGHT ) ) {
duration += 2_hours;
}
p->add_effect( effect_valium, duration );
p->add_effect( effect_took_anticonvulsant_visible, duration );
p->add_effect( effect_high, duration );
if( p->has_effect( effect_shakes ) ) {
p->remove_effect( effect_shakes );
p->add_msg_if_player( m_good, _( "You stop shaking." ) );
}
return it->type->charges_to_use();
}
int iuse::weed_cake( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player(
_( "You start scarfing down the delicious cake. It tastes a little funny though..." ) );
time_duration duration = 12_minutes;
if( p->has_trait( trait_TOLERANCE ) ) {
duration = 9_minutes;
}
if( p->has_trait( trait_LIGHTWEIGHT ) ) {
duration = 15_minutes;
}
p->mod_hunger( 2 );
p->mod_thirst( 6 );
if( p->get_painkiller() < 5 ) {
p->set_painkiller( ( p->get_painkiller() + 3 ) * 2 );
}
p->add_effect( effect_weed_high, duration );
p->moves -= 100;
if( one_in( 5 ) ) {
weed_msg( *p );
}
return it->type->charges_to_use();
}
int iuse::coke( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You snort a bump of coke." ) );
/** @EFFECT_STR reduces duration of coke */
time_duration duration = 20_minutes - 1_seconds * p->str_cur + rng( 0_minutes, 1_minutes );
if( p->has_trait( trait_TOLERANCE ) ) {
duration -= 1_minutes; // Symmetry would cause problems :-/
}
if( p->has_trait( trait_LIGHTWEIGHT ) ) {
duration += 2_minutes;
}
p->mod_hunger( -8 );
p->add_effect( effect_high, duration );
return it->type->charges_to_use();
}
int iuse::meth( player *p, item *it, bool, const tripoint & )
{
/** @EFFECT_STR reduces duration of meth */
time_duration duration = 1_minutes * ( 60 - p->str_cur );
if( p->has_amount( "apparatus", 1 ) && p->use_charges_if_avail( "fire", 1 ) ) {
p->add_msg_if_player( m_neutral, _( "You smoke your meth." ) );
p->add_msg_if_player( m_good, _( "The world seems to sharpen." ) );
p->mod_fatigue( -375 );
if( p->has_trait( trait_TOLERANCE ) ) {
duration *= 1.2;
} else {
duration *= ( p->has_trait( trait_LIGHTWEIGHT ) ? 1.8 : 1.5 );
}
// breathe out some smoke
for( int i = 0; i < 3; i++ ) {
g->m.add_field( {p->posx() + static_cast<int>( rng( -2, 2 ) ), p->posy() + static_cast<int>( rng( -2, 2 ) ), p->posz()},
fd_methsmoke, 2 );
}
} else {
p->add_msg_if_player( _( "You snort some crystal meth." ) );
p->mod_fatigue( -300 );
}
if( !p->has_effect( effect_meth ) ) {
duration += 1_hours;
}
if( duration > 0_turns ) {
// meth actually inhibits hunger, weaker characters benefit more
/** @EFFECT_STR_MAX >4 experiences less hunger benefit from meth */
int hungerpen = ( p->str_max < 5 ? 35 : 40 - ( 2 * p->str_max ) );
if( hungerpen > 0 ) {
p->mod_hunger( -hungerpen );
}
p->add_effect( effect_meth, duration );
}
return it->type->charges_to_use();
}
int iuse::vaccine( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You inject the vaccine." ) );
p->add_msg_if_player( m_good, _( "You feel tough." ) );
p->mod_healthy_mod( 200, 200 );
p->mod_pain( 3 );
item syringe( "syringe", it->birthday() );
p->i_add( syringe );
return it->type->charges_to_use();
}
int iuse::flu_vaccine( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You inject the vaccine." ) );
p->add_msg_if_player( m_good, _( "You no longer need to fear the flu." ) );
p->add_effect( effect_flushot, 1_turns, num_bp, true );
p->mod_pain( 3 );
item syringe( "syringe", it->birthday() );
p->i_add( syringe );
return it->type->charges_to_use();
}
int iuse::poison( player *p, item *it, bool, const tripoint & )
{
if( ( p->has_trait( trait_EATDEAD ) ) ) {
return it->type->charges_to_use();
}
// NPCs have a magical sense of what is inedible
// Players can abuse the crafting menu instead...
if( !it->has_flag( "HIDDEN_POISON" ) &&
( p->is_npc() ||
!p->query_yn( _( "Are you sure you want to eat this? It looks poisonous..." ) ) ) ) {
return 0;
}
/** @EFFECT_STR increases EATPOISON trait effectiveness (50-90%) */
if( ( p->has_trait( trait_EATPOISON ) ) && ( !( one_in( p->str_cur / 2 ) ) ) ) {
return it->type->charges_to_use();
}
p->add_effect( effect_poison, 1_hours );
p->add_effect( effect_foodpoison, 3_hours );
return it->type->charges_to_use();
}
int iuse::meditate( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
if( p->has_trait( trait_SPIRITUAL ) ) {
p->assign_activity( activity_id( "ACT_MEDITATE" ), 2000 );
} else {
p->add_msg_if_player( _( "This %s probably meant a lot to someone at one time." ),
it->tname() );
}
return it->type->charges_to_use();
}
int iuse::thorazine( player *p, item *it, bool, const tripoint & )
{
p->mod_fatigue( 5 );
p->remove_effect( effect_hallu );
p->remove_effect( effect_visuals );
p->remove_effect( effect_high );
if( !p->has_effect( effect_dermatik ) ) {
p->remove_effect( effect_formication );
}
if( one_in( 50 ) ) { // adverse reaction
p->add_msg_if_player( m_bad, _( "You feel completely exhausted." ) );
p->mod_fatigue( 15 );
} else {
p->add_msg_if_player( m_warning, _( "You feel a bit wobbly." ) );
}
return it->type->charges_to_use();
}
int iuse::prozac( player *p, item *it, bool, const tripoint & )
{
if( !p->has_effect( effect_took_prozac ) ) {
p->add_effect( effect_took_prozac, 12_hours );
} else {
p->stim += 3;
}
if( one_in( 50 ) ) { // adverse reaction, same duration as prozac effect.
p->add_msg_if_player( m_warning, _( "You suddenly feel hollow inside." ) );
p->add_effect( effect_took_prozac_bad, p->get_effect_dur( effect_took_prozac ) );
}
p->add_effect( effect_took_prozac_visible, rng( 9_hours, 15_hours ) );
return it->type->charges_to_use();
}
int iuse::sleep( player *p, item *it, bool, const tripoint & )
{
p->mod_fatigue( 40 );
p->add_msg_if_player( m_warning, _( "You feel very sleepy..." ) );
return it->type->charges_to_use();
}
int iuse::datura( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
return 0;
}
p->add_effect( effect_datura, rng( 3_hours, 13_hours ) );
p->add_msg_if_player( _( "You eat the datura seed." ) );
if( p->has_trait( trait_SPIRITUAL ) ) {
p->add_morale( MORALE_FOOD_GOOD, 36, 72, 2_hours, 1_hours, false, it->type );
}
return it->type->charges_to_use();
}
int iuse::flumed( player *p, item *it, bool, const tripoint & )
{
p->add_effect( effect_took_flumed, 10_hours );
p->add_msg_if_player( _( "You take some %s" ), it->tname() );
return it->type->charges_to_use();
}
int iuse::flusleep( player *p, item *it, bool, const tripoint & )
{
p->add_effect( effect_took_flumed, 12_hours );
p->mod_fatigue( 30 );
p->add_msg_if_player( _( "You take some %s" ), it->tname() );
p->add_msg_if_player( m_warning, _( "You feel very sleepy..." ) );
return it->type->charges_to_use();
}
int iuse::inhaler( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( m_neutral, _( "You take a puff from your inhaler." ) );
if( !p->remove_effect( effect_asthma ) ) {
p->mod_fatigue( -3 ); // if we don't have asthma can be used as stimulant
if( one_in( 20 ) ) { // with a small but significant risk of adverse reaction
p->add_effect( effect_shakes, rng( 2_minutes, 5_minutes ) );
}
}
p->remove_effect( effect_smoke );
return it->type->charges_to_use();
}
int iuse::oxygen_bottle( player *p, item *it, bool, const tripoint & )
{
p->moves -= to_moves<int>( 10_seconds );
p->add_msg_if_player( m_neutral, _( "You breathe deeply from the %s" ), it->tname() );
if( p->has_effect( effect_smoke ) ) {
p->remove_effect( effect_smoke );
} else if( p->has_effect( effect_teargas ) ) {
p->remove_effect( effect_teargas );
} else if( p->has_effect( effect_asthma ) ) {
p->remove_effect( effect_asthma );
} else if( p->stim < 16 ) {
p->stim += 8;
p->mod_painkiller( 2 );
}
p->mod_painkiller( 2 );
return it->type->charges_to_use();
}
int iuse::blech( player *p, item *it, bool, const tripoint & )
{
// TODO: Add more effects?
if( it->made_of( LIQUID ) ) {
if( !p->query_yn( _( "This looks unhealthy, sure you want to drink it?" ) ) ) {
return 0;
}
} else { //Assume that if a blech consumable isn't a drink, it will be eaten.
if( !p->query_yn( _( "This looks unhealthy, sure you want to eat it?" ) ) ) {
return 0;
}
}
if( it->has_flag( "ACID" ) && ( p->has_trait( trait_ACIDPROOF ) ||
p->has_trait( trait_ACIDBLOOD ) ) ) {
p->add_msg_if_player( m_bad, _( "Blech, that tastes gross!" ) );
//reverse the harmful values of drinking this acid.
double multiplier = -1;
p->stomach.mod_nutr( -p->nutrition_for( *it ) * multiplier );
p->mod_thirst( -it->get_comestible()->quench * multiplier );
p->stomach.mod_quench( 20 ); //acidproof people can drink acids like diluted water.
p->mod_healthy_mod( it->get_comestible()->healthy * multiplier,
it->get_comestible()->healthy * multiplier );
p->add_morale( MORALE_FOOD_BAD, it->get_comestible()->fun * multiplier, 60, 1_hours, 30_minutes,
false, it->type );
} else {
p->add_msg_if_player( m_bad, _( "Blech, that burns your throat!" ) );
p->mod_pain( rng( 32, 64 ) );
p->add_effect( effect_poison, 1_hours );
p->apply_damage( nullptr, bp_torso, rng( 4, 12 ) );
p->vomit();
}
return it->type->charges_to_use();
}
int iuse::plantblech( player *p, item *it, bool, const tripoint &pos )
{
if( p->has_trait( trait_THRESH_PLANT ) ) {
double multiplier = -1;
if( p->has_trait( trait_CHLOROMORPH ) ) {
multiplier = -3;
p->add_msg_if_player( m_good, _( "The meal is revitalizing." ) );
} else {
p->add_msg_if_player( m_good, _( "Oddly enough, this doesn't taste so bad." ) );
}
//reverses the harmful values of drinking fertilizer
p->stomach.mod_nutr( p->nutrition_for( *it ) * multiplier );
p->mod_thirst( -it->get_comestible()->quench * multiplier );
p->mod_healthy_mod( it->get_comestible()->healthy * multiplier,
it->get_comestible()->healthy * multiplier );
p->add_morale( MORALE_FOOD_GOOD, -10 * multiplier, 60, 1_hours, 30_minutes, false, it->type );
return it->type->charges_to_use();
} else {
return blech( p, it, true, pos );
}
}
int iuse::chew( player *p, item *it, bool, const tripoint & )
{
// TODO: Add more effects?
p->add_msg_if_player( _( "You chew your %s." ), it->tname() );
return it->type->charges_to_use();
}
// Helper to handle the logic of removing some random mutations.
static void do_purify( player &p )
{
std::vector<trait_id> valid; // Which flags the player has
for( auto &traits_iter : mutation_branch::get_all() ) {
if( p.has_trait( traits_iter.id ) && !p.has_base_trait( traits_iter.id ) ) {
//Looks for active mutation
valid.push_back( traits_iter.id );
}
}
if( valid.empty() ) {
p.add_msg_if_player( _( "You feel cleansed." ) );
return;
}
int num_cured = rng( 1, valid.size() );
num_cured = std::min( 4, num_cured );
for( int i = 0; i < num_cured && !valid.empty(); i++ ) {
const trait_id id = random_entry_removed( valid );
if( p.purifiable( id ) ) {
p.remove_mutation( id );
} else {
p.add_msg_if_player( m_warning, _( "You feel a slight itching inside, but it passes." ) );
}
}
}
int iuse::purifier( player *p, item *it, bool, const tripoint & )
{
mutagen_attempt checks = mutagen_common_checks( *p, *it, false,
pgettext( "memorial_male", "Consumed purifier." ), pgettext( "memorial_female",
"Consumed purifier." ) );
if( !checks.allowed ) {
return checks.charges_used;
}
do_purify( *p );
return it->type->charges_to_use();
}
int iuse::purify_iv( player *p, item *it, bool, const tripoint & )
{
mutagen_attempt checks = mutagen_common_checks( *p, *it, false,
pgettext( "memorial_male", "Injected purifier." ), pgettext( "memorial_female",
"Injected purifier." ) );
if( !checks.allowed ) {
return checks.charges_used;
}
std::vector<trait_id> valid; // Which flags the player has
for( auto &traits_iter : mutation_branch::get_all() ) {
if( p->has_trait( traits_iter.id ) && !p->has_base_trait( traits_iter.id ) ) {
//Looks for active mutation
valid.push_back( traits_iter.id );
}
}
if( valid.empty() ) {
p->add_msg_if_player( _( "You feel cleansed." ) );
return it->type->charges_to_use();
}
int num_cured = rng( 4,
valid.size() ); //Essentially a double-strength purifier, but guaranteed at least 4. Double-edged and all
if( num_cured > 8 ) {
num_cured = 8;
}
for( int i = 0; i < num_cured && !valid.empty(); i++ ) {
const trait_id id = random_entry_removed( valid );
if( p->purifiable( id ) ) {
p->remove_mutation( id );
} else {
p->add_msg_if_player( m_warning, _( "You feel a distinct burning inside, but it passes." ) );
}
if( !( p->has_trait( trait_NOPAIN ) ) ) {
p->mod_pain( 2 * num_cured ); //Hurts worse as it fixes more
p->add_msg_if_player( m_warning, _( "Feels like you're on fire, but you're OK." ) );
}
p->mod_stored_nutr( 2 * num_cured );
p->mod_thirst( 2 * num_cured );
p->mod_fatigue( 2 * num_cured );
}
return it->type->charges_to_use();
}
int iuse::purify_smart( player *p, item *it, bool, const tripoint & )
{
mutagen_attempt checks = mutagen_common_checks( *p, *it, false,
pgettext( "memorial_male", "Injected smart purifier." ), pgettext( "memorial_female",
"Injected smart purifier." ) );
if( !checks.allowed ) {
return checks.charges_used;
}
std::vector<trait_id> valid; // Which flags the player has
std::vector<std::string> valid_names; // Which flags the player has
for( auto &traits_iter : mutation_branch::get_all() ) {
if( p->has_trait( traits_iter.id ) &&
!p->has_base_trait( traits_iter.id ) &&
p->purifiable( traits_iter.id ) ) {
//Looks for active mutation
valid.push_back( traits_iter.id );
valid_names.push_back( traits_iter.id->name() );
}
}
if( valid.empty() ) {
p->add_msg_if_player( _( "You don't have any mutations to purify." ) );
return 0;
}
int mutation_index = uilist( _( "Choose a mutation to purify" ), valid_names );
if( mutation_index < 0 ) {
return 0;
}
p->add_msg_if_player(
_( "You inject the purifier. The liquid thrashes inside the tube and goes down reluctantly." ) );
p->remove_mutation( valid[mutation_index] );
valid.erase( valid.begin() + mutation_index );
// and one or two more untargeted purifications.
if( !valid.empty() ) {
p->remove_mutation( random_entry_removed( valid ) );
}
if( !valid.empty() && one_in( 2 ) ) {
p->remove_mutation( random_entry_removed( valid ) );
}
p->mod_pain( 3 );
item syringe( "syringe", it->birthday() );
p->i_add( syringe );
return it->type->charges_to_use();
}
static void spawn_spores( const player &p )
{
int spores_spawned = 0;
fungal_effects fe( *g, g->m );
for( const tripoint &dest : closest_tripoints_first( 4, p.pos() ) ) {
if( g->m.impassable( dest ) ) {
continue;
}
float dist = rl_dist( dest, p.pos() );
if( x_in_y( 1, dist ) ) {
fe.marlossify( dest );
}
if( g->critter_at( dest ) != nullptr ) {
continue;
}
if( one_in( 10 + 5 * dist ) && one_in( spores_spawned * 2 ) ) {
if( monster *const spore = g->summon_mon( mon_spore, dest ) ) {
spore->friendly = -1;
spores_spawned++;
}
}
}
}
static void marloss_common( player &p, item &it, const trait_id &current_color )
{
static const std::map<trait_id, add_type> mycus_colors = {{
{ trait_MARLOSS_BLUE, ADD_MARLOSS_B }, { trait_MARLOSS_YELLOW, ADD_MARLOSS_Y }, { trait_MARLOSS, ADD_MARLOSS_R }
}
};
if( p.has_trait( current_color ) || p.has_trait( trait_THRESH_MARLOSS ) ) {
p.add_msg_if_player( m_good,
_( "As you eat the %s, you have a near-religious experience, feeling at one with your surroundings..." ),
it.tname() );
p.add_morale( MORALE_MARLOSS, 100, 1000 );
for( const std::pair<trait_id, add_type> &pr : mycus_colors ) {
if( pr.first != current_color ) {
p.add_addiction( pr.second, 50 );
}
}
p.set_hunger( -10 );
spawn_spores( p );
return;
}
int marloss_count = std::count_if( mycus_colors.begin(), mycus_colors.end(),
[&p]( const std::pair<trait_id, add_type> &pr ) {
return p.has_trait( pr.first );
} );
/* If we're not already carriers of current type of Marloss, roll for a random effect:
* 1 - Mutate
* 2 - Mutate
* 3 - Mutate
* 4 - Purify
* 5 - Purify
* 6 - Cleanse radiation + Purify
* 7 - Fully satiate
* 8 - Vomit
* 9-12 - Give Marloss mutation
*/
int effect = rng( 1, 12 );
if( effect <= 3 ) {
p.add_msg_if_player( _( "It tastes extremely strange!" ) );
p.mutate();
// Gruss dich, mutation drain, missed you!
p.mod_pain( 2 * rng( 1, 5 ) );
p.mod_stored_nutr( 10 );
p.mod_thirst( 10 );
p.mod_fatigue( 5 );
} else if( effect <= 6 ) { // Radiation cleanse is below
p.add_msg_if_player( m_good, _( "You feel better all over." ) );
p.mod_painkiller( 30 );
iuse dummy;
dummy.purifier( &p, &it, false, p.pos() );
if( effect == 6 ) {
p.radiation = 0;
}
} else if( effect == 7 ) {
// previously used to set hunger to -10. with the new system, needs to do something
// else that actually makes sense, so it is a little bit more involved.
units::volume fulfill_vol = std::max( p.stomach.capacity() / 8 - p.stomach.contains(), 0_ml );
if( fulfill_vol != 0_ml ) {
p.add_msg_if_player( m_good, _( "It is delicious, and very filling!" ) );
int fulfill_cal = units::to_milliliter( fulfill_vol * 6 );
p.stomach.mod_calories( fulfill_cal );
p.stomach.mod_contents( fulfill_vol );
} else {
p.add_msg_if_player( m_bad, _( "It is delicious, but you can't eat any more." ) );
}
} else if( effect == 8 ) {
p.add_msg_if_player( m_bad, _( "You take one bite, and immediately vomit!" ) );
p.vomit();
} else if( p.crossed_threshold() ) {
// Mycus Rejection. Goo already present fights off the fungus.
p.add_msg_if_player( m_bad,
_( "You feel a familiar warmth, but suddenly it surges into an excruciating burn as you convulse, vomiting, and black out..." ) );
p.add_memorial_log( pgettext( "memorial_male", "Suffered Marloss Rejection." ),
pgettext( "memorial_female", "Suffered Marloss Rejection." ) );
p.vomit();
p.mod_pain( 90 );
p.hurtall( rng( 40, 65 ), nullptr ); // No good way to say "lose half your current HP"
/** @EFFECT_INT slightly reduces sleep duration when eating mycus+goo */
p.fall_asleep( 10_hours - p.int_cur *
1_minutes ); // Hope you were eating someplace safe. Mycus v. Goo in your guts is no joke.
for( const std::pair<trait_id, add_type> &pr : mycus_colors ) {
p.unset_mutation( pr.first );
p.rem_addiction( pr.second );
}
p.set_mutation(
trait_MARLOSS_AVOID ); // And if you survive it's etched in your RNA, so you're unlikely to repeat the experiment.
} else if( marloss_count >= 2 ) {
p.add_msg_if_player( m_bad,
_( "You feel a familiar warmth, but suddenly it surges into painful burning as you convulse and collapse to the ground..." ) );
/** @EFFECT_INT reduces sleep duration when eating wrong color marloss */
p.fall_asleep( 40_minutes - 1_minutes * p.int_cur / 2 );
for( const std::pair<trait_id, add_type> &pr : mycus_colors ) {
p.unset_mutation( pr.first );
p.rem_addiction( pr.second );
}
p.set_mutation( trait_THRESH_MARLOSS );
g->m.ter_set( p.pos(), t_marloss );
p.add_memorial_log( pgettext( "memorial_male", "Opened the Marloss Gateway." ),
pgettext( "memorial_female", "Opened the Marloss Gateway." ) );
p.add_msg_if_player( m_good,
_( "You wake up in a marloss bush. Almost *cradled* in it, actually, as though it grew there for you." ) );
//~ Beginning to hear the Mycus while conscious: that's it speaking
p.add_msg_if_player( m_good,
_( "unity. together we have reached the door. we provide the final key. now to pass through..." ) );
} else {
p.add_msg_if_player( _( "You feel a strange warmth spreading throughout your body..." ) );
p.set_mutation( current_color );
// Give us addictions to the other two colors, but cure one for current color
for( const std::pair<trait_id, add_type> &pr : mycus_colors ) {
if( pr.first == current_color ) {
p.rem_addiction( pr.second );
} else {
p.add_addiction( pr.second, 60 );
}
}
}
}
static bool marloss_prevented( const player &p )
{
if( p.is_npc() ) {
return true;
}
if( p.has_trait( trait_MARLOSS_AVOID ) ) {
//~"Uh-uh" is a sound used for "nope", "no", etc.
p.add_msg_if_player( m_warning,
_( "After what happened that last time? uh-uh. You're not eating that alien poison." ) );
return true;
}
if( p.has_trait( trait_THRESH_MYCUS ) ) {
p.add_msg_if_player( m_info,
_( "We no longer require this scaffolding. We reserve it for other uses." ) );
return true;
}
return false;
}
int iuse::marloss( player *p, item *it, bool, const tripoint & )
{
if( marloss_prevented( *p ) ) {
return 0;
}
p->add_memorial_log( pgettext( "memorial_male", "Ate a marloss berry." ),
pgettext( "memorial_female", "Ate a marloss berry." ) );
marloss_common( *p, *it, trait_MARLOSS );
return it->type->charges_to_use();
}
int iuse::marloss_seed( player *p, item *it, bool, const tripoint & )
{
if( !query_yn( _( "Sure you want to eat the %s? You could plant it in a mound of dirt." ),
colorize( it->tname(), it->color_in_inventory() ) ) ) {
return 0; // Save the seed for later!
}
if( marloss_prevented( *p ) ) {
return 0;
}
p->add_memorial_log( pgettext( "memorial_male", "Ate a marloss seed." ),
pgettext( "memorial_female", "Ate a marloss seed." ) );
marloss_common( *p, *it, trait_MARLOSS_BLUE );
return it->type->charges_to_use();
}
int iuse::marloss_gel( player *p, item *it, bool, const tripoint & )
{
if( marloss_prevented( *p ) ) {
return 0;
}
p->add_memorial_log( pgettext( "memorial_male", "Ate some marloss jelly." ),
pgettext( "memorial_female", "Ate some marloss jelly." ) );
marloss_common( *p, *it, trait_MARLOSS_YELLOW );
return it->type->charges_to_use();
}
int iuse::mycus( player *p, item *it, bool t, const tripoint &pos )
{
if( p->is_npc() ) {
return it->type->charges_to_use();
}
// Welcome our guide. Welcome. To. The Mycus.
if( p->has_trait( trait_THRESH_MARLOSS ) ) {
p->add_memorial_log( pgettext( "memorial_male", "Became one with the Mycus." ),
pgettext( "memorial_female", "Became one with the Mycus." ) );
p->add_msg_if_player( m_neutral,
_( "The apple tastes amazing, and you finish it quickly, not even noticing the lack of any core or seeds." ) );
p->add_msg_if_player( m_good, _( "You feel better all over." ) );
p->mod_painkiller( 30 );
this->purifier( p, it, t, pos ); // Clear out some of that goo you may have floating around
p->radiation = 0;
p->healall( 4 ); // Can't make you a whole new person, but not for lack of trying
p->add_msg_if_player( m_good,
_( "As the apple settles in, you feel ecstasy radiating through every part of your body..." ) );
p->add_morale( MORALE_MARLOSS, 1000, 1000 ); // Last time you'll ever have it this good. So enjoy.
p->add_msg_if_player( m_good,
_( "Your eyes roll back in your head. Everything dissolves into a blissful haze..." ) );
/** @EFFECT_INT slightly reduces sleep duration when eating mycus */
p->fall_asleep( 5_hours - p->int_cur * 1_minutes );
p->unset_mutation( trait_THRESH_MARLOSS );
p->set_mutation( trait_THRESH_MYCUS );
//~ The Mycus does not use the term (or encourage the concept of) "you". The PC is a local/native organism, but is now the Mycus.
//~ It still understands the concept, but uninitelligent fungaloids and mind-bent symbiotes should not need it.
//~ We are the Mycus.
g->refresh_all();
popup( _( "We welcome into us. We have endured long in this forbidding world." ) );
p->add_msg_if_player( " " );
p->add_msg_if_player( m_good,
_( "A sea of white caps, waving gently. A haze of spores wafting silently over a forest." ) );
g->refresh_all();
popup( _( "The natives have a saying: \"E Pluribus Unum.\" Out of many, one." ) );
p->add_msg_if_player( " " );
p->add_msg_if_player( m_good,
_( "The blazing pink redness of the berry. The juices spreading across your tongue, the warmth draping over us like a lover's embrace." ) );
g->refresh_all();
popup( _( "We welcome the union of our lines in our local guide. We will prosper, and unite this world. Even now, our fruits adapt to better serve local physiology." ) );
p->add_msg_if_player( " " );
p->add_msg_if_player( m_good,
_( "The sky-blue of the seed. The nutty, creamy flavors intermingling with the berry, a memory that will never leave us." ) );
g->refresh_all();
popup( _( "As, in time, shall we adapt to better welcome those who have not received us." ) );
p->add_msg_if_player( " " );
p->add_msg_if_player( m_good,
_( "The amber-yellow of the sap. Feel it flowing through our veins, taking the place of the strange, thin red gruel called \"blood.\"" ) );
g->refresh_all();
popup( _( "We are the Mycus." ) );
/*p->add_msg_if_player( m_good,
_( "We welcome into us. We have endured long in this forbidding world." ) );
p->add_msg_if_player( m_good,
_( "The natives have a saying: \"E Pluribus Unum\" Out of many, one." ) );
p->add_msg_if_player( m_good,
_( "We welcome the union of our lines in our local guide. We will prosper, and unite this world." ) );
p->add_msg_if_player( m_good, _( "Even now, our fruits adapt to better serve local physiology." ) );
p->add_msg_if_player( m_good,
_( "As, in time, shall we adapt to better welcome those who have not received us." ) );*/
fungal_effects fe( *g, g->m );
for( const tripoint &pos : g->m.points_in_radius( p->pos(), 3 ) ) {
fe.marlossify( pos );
}
p->rem_addiction( ADD_MARLOSS_R );
p->rem_addiction( ADD_MARLOSS_B );
p->rem_addiction( ADD_MARLOSS_Y );
} else if( p->has_trait( trait_THRESH_MYCUS ) &&
!p->has_trait( trait_M_DEPENDENT ) ) { // OK, now set the hook.
if( !one_in( 3 ) ) {
p->mutate_category( "MYCUS" );
p->mod_stored_nutr( 10 );
p->mod_thirst( 10 );
p->mod_fatigue( 5 );
p->add_morale( MORALE_MARLOSS, 25, 200 ); // still covers up mutation pain
}
} else if( p->has_trait( trait_THRESH_MYCUS ) ) {
p->mod_painkiller( 5 );
p->stim += 5;
} else { // In case someone gets one without having been adapted first.
// Marloss is the Mycus' method of co-opting humans. Mycus fruit is for symbiotes' maintenance and development.
p->add_msg_if_player(
_( "This apple tastes really weird! You're not sure it's good for you..." ) );
p->mutate();
p->mod_pain( 2 * rng( 1, 5 ) );
p->mod_stored_nutr( 10 );
p->mod_thirst( 10 );
p->mod_fatigue( 5 );
p->vomit(); // no hunger/quench benefit for you
p->mod_healthy_mod( -8, -50 );
}
return it->type->charges_to_use();
}
// Types of petfood for taming each different monster.
enum Petfood {
DOGFOOD,
CATFOOD,
CATTLEFODDER,
BIRDFOOD
};
static int feedpet( player &p, monster &mon, item &it, m_flag food_flag, const char *message )
{
if( mon.has_flag( food_flag ) ) {
p.add_msg_if_player( m_good, message, mon.get_name() );
mon.friendly = -1;
mon.add_effect( effect_pet, 1_turns, num_bp, true );
p.consume_charges( it, 1 );
return 1;
} else {
p.add_msg_if_player( _( "The %s doesn't want that kind of food." ), mon.get_name() );
return 0;
}
}
static int petfood( player &p, item &it, Petfood animal_food_type )
{
const cata::optional<tripoint> pnt_ = choose_adjacent( string_format( _( "Put the %s where?" ),
it.tname() ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
p.moves -= to_moves<int>( 1_seconds );
// First a check to see if we are trying to feed a NPC dog food.
if( animal_food_type == DOGFOOD && g->critter_at<npc>( pnt ) != nullptr ) {
if( npc *const person_ = g->critter_at<npc>( pnt ) ) {
npc &person = *person_;
if( query_yn( _( "Are you sure you want to feed a person the dog food?" ) ) ) {
p.add_msg_if_player( _( "You put your %1$s into %2$s's mouth!" ), it.tname(),
person.name );
if( person.is_ally( p ) || x_in_y( 9, 10 ) ) {
person.say(
_( "Okay, but please, don't give me this again. I don't want to eat dog food in the cataclysm all day." ) );
p.consume_charges( it, 1 );
return 1;
} else {
p.add_msg_if_player( _( "%s knocks it out from your hand!" ), person.name );
person.make_angry();
p.consume_charges( it, 1 );
return 1;
}
} else {
p.add_msg_if_player( _( "Never mind." ) );
return 0;
}
}
// Then monsters.
} else if( monster *const mon_ptr = g->critter_at<monster>( pnt, true ) ) {
monster &mon = *mon_ptr;
if( mon.is_hallucination() ) {
p.add_msg_if_player( _( "You try to feed the %s some %s, but it vanishes!" ),
mon.type->nname(), it.tname() );
mon.die( nullptr );
return 0;
}
// This switch handles each petfood for each type of tameable monster.
switch( animal_food_type ) {
case DOGFOOD:
if( mon.type->id == mon_dog_thing ) {
p.deal_damage( &mon, bp_hand_r, damage_instance( DT_CUT, rng( 1, 10 ) ) );
p.add_msg_if_player( m_bad, _( "You want to feed it the dog food, but it bites your fingers!" ) );
if( one_in( 5 ) ) {
p.add_msg_if_player(
_( "Apparently it's more interested in your flesh than the dog food in your hand!" ) );
p.consume_charges( it, 1 );
return 1;
}
} else {
return feedpet( p, mon, it, MF_DOGFOOD,
_( "The %s seems to like you! It lets you pat its head and seems friendly." ) );
}
break;
case CATFOOD:
return feedpet( p, mon, it, MF_CATFOOD,
_( "The %s seems to like you! Or maybe it just tolerates your presence better. It's hard to tell with felines." ) );
case CATTLEFODDER:
return feedpet( p, mon, it, MF_CATTLEFODDER,
_( "The %s seems to like you! It lets you pat its head and seems friendly." ) );
case BIRDFOOD:
return feedpet( p, mon, it, MF_BIRDFOOD,
_( "The %s seems to like you! It runs around your legs and seems friendly." ) );
}
} else {
p.add_msg_if_player( _( "There is nothing to be fed here." ) );
return 0;
}
return 1;
}
int iuse::dogfood( player *p, item *it, bool, const tripoint & )
{
return petfood( *p, *it, DOGFOOD );
}
int iuse::catfood( player *p, item *it, bool, const tripoint & )
{
return petfood( *p, *it, CATFOOD );
}
int iuse::feedcattle( player *p, item *it, bool, const tripoint & )
{
return petfood( *p, *it, CATTLEFODDER );
}
int iuse::feedbird( player *p, item *it, bool, const tripoint & )
{
return petfood( *p, *it, BIRDFOOD );
}
int iuse::sew_advanced( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( p->fine_detail_vision_mod() > 4 ) {
add_msg( m_info, _( "You can't see to sew!" ) );
return 0;
}
static const std::set<material_id> sewable{
material_id( "cotton" ),
material_id( "leather" ),
material_id( "fur" ),
material_id( "nomex" ),
material_id( "plastic" ),
material_id( "kevlar" ),
material_id( "wool" )
};
int pos = g->inv_for_filter( _( "Enhance which clothing?" ), []( const item & itm ) {
return itm.is_armor() && !itm.is_firearm() && !itm.is_power_armor() &&
itm.made_of_any( sewable );
} );
item &mod = p->i_at( pos );
if( mod.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return 0;
}
if( &mod == it ) {
p->add_msg_if_player( m_info,
_( "This can be used to repair or modify other items, not itself." ) );
return 0;
}
// Gives us an item with the mod added or removed (toggled)
const auto modded_copy = []( const item & proto, const std::string & mod_type ) {
item mcopy = proto;
if( mcopy.item_tags.count( mod_type ) == 0 ) {
mcopy.item_tags.insert( mod_type );
} else {
mcopy.item_tags.erase( mod_type );
}
return mcopy;
};
// TODO: Wrap all the mods into structs, maybe even json-able
// All possible mods here
std::array<std::string, 4> clothing_mods{
{ "wooled", "furred", "leather_padded", "kevlar_padded" }
};
// Materials those mods use
std::array<std::string, 4> mod_materials{
{ "felt_patch", "fur", "leather", "kevlar_plate" }
};
// Cache available materials
std::map< itype_id, bool > has_enough;
const int items_needed = mod.volume() / 750_ml + 1;
const inventory &crafting_inv = p->crafting_inventory();
// Go through all discovered repair items and see if we have any of them available
for( auto &material : mod_materials ) {
has_enough[material] = crafting_inv.has_amount( material, items_needed );
}
const int mod_count = mod.item_tags.count( "wooled" ) + mod.item_tags.count( "furred" ) +
mod.item_tags.count( "leather_padded" ) + mod.item_tags.count( "kevlar_padded" );
// We need extra thread to lose it on bad rolls
const int thread_needed = mod.volume() / 125_ml + 10;
// Returns true if the item already has the mod or if we have enough materials and thread to add it
const auto can_add_mod = [&]( const std::string & new_mod, const itype_id & mat_item ) {
return mod.item_tags.count( new_mod ) > 0 ||
( it->charges >= thread_needed && has_enough[mat_item] );
};
uilist tmenu;
// TODO: Tell how much thread will we use
if( it->charges >= thread_needed ) {
tmenu.text = _( "How do you want to modify it?" );
} else {
tmenu.text = _( "Not enough thread to modify. Which modification do you want to remove?" );
}
// TODO: The supremely ugly block of code below looks better than 200 line boilerplate
// that was there before, but it can probably be moved into a helper somehow
// TODO: 2: List how much material we have and how much we need
item temp_item = modded_copy( mod, "wooled" );
// Can we perform this addition or removal
bool enab = can_add_mod( "wooled", "felt_patch" );
tmenu.addentry( 0, enab, MENU_AUTOASSIGN, _( "%s (Warmth: %d->%d, Encumbrance: %d->%d)" ),
mod.item_tags.count( "wooled" ) == 0 ? _( "Line it with wool" ) : _( "Destroy wool lining" ),
mod.get_warmth(), temp_item.get_warmth(), mod.get_encumber( *p ), temp_item.get_encumber( *p ) );
temp_item = modded_copy( mod, "furred" );
enab = can_add_mod( "furred", "fur" );
tmenu.addentry( 1, enab, MENU_AUTOASSIGN, _( "%s (Warmth: %d->%d, Encumbrance: %d->%d)" ),
mod.item_tags.count( "furred" ) == 0 ? _( "Line it with fur" ) : _( "Destroy fur lining" ),
mod.get_warmth(), temp_item.get_warmth(), mod.get_encumber( *p ), temp_item.get_encumber( *p ) );
temp_item = modded_copy( mod, "leather_padded" );
enab = can_add_mod( "leather_padded", "leather" );
tmenu.addentry( 2, enab, MENU_AUTOASSIGN, _( "%s (Bash/Cut: %d/%d->%d/%d, Encumbrance: %d->%d)" ),
mod.item_tags.count( "leather_padded" ) == 0 ? _( "Pad with leather" ) :
_( "Destroy leather padding" ),
mod.bash_resist(), mod.cut_resist(), temp_item.bash_resist(), temp_item.cut_resist(),
mod.get_encumber( *p ), temp_item.get_encumber( *p ) );
temp_item = modded_copy( mod, "kevlar_padded" );
enab = can_add_mod( "kevlar_padded", "kevlar_plate" );
tmenu.addentry( 3, enab, MENU_AUTOASSIGN, _( "%s (Bash/Cut: %d/%d->%d/%d, Encumbrance: %d->%d)" ),
mod.item_tags.count( "kevlar_padded" ) == 0 ? _( "Pad with Kevlar" ) :
_( "Destroy Kevlar padding" ),
mod.bash_resist(), mod.cut_resist(), temp_item.bash_resist(), temp_item.cut_resist(),
mod.get_encumber( *p ), temp_item.get_encumber( *p ) );
tmenu.query();
const int choice = tmenu.ret;
if( choice < 0 || choice > 3 ) {
return 0;
}
// The mod player picked
const std::string &the_mod = clothing_mods[choice];
// If the picked mod already exists, player wants to destroy it
if( mod.item_tags.count( the_mod ) ) {
if( query_yn( _( "Are you sure? You will not gain any materials back." ) ) ) {
mod.item_tags.erase( the_mod );
}
return 0;
}
// Get the id of the material used
const auto &repair_item = mod_materials[choice];
std::vector<item_comp> comps;
comps.push_back( item_comp( repair_item, items_needed ) );
p->moves -= to_moves<int>( 30_seconds * p->fine_detail_vision_mod() );
p->practice( skill_tailor, items_needed * 3 + 3 );
/** @EFFECT_TAILOR randomly improves clothing modification efforts */
int rn = dice( 3, 2 + p->get_skill_level( skill_tailor ) ); // Skill
/** @EFFECT_DEX randomly improves clothing modification efforts */
rn += rng( 0, p->dex_cur / 2 ); // Dexterity
/** @EFFECT_PER randomly improves clothing modification efforts */
rn += rng( 0, p->per_cur / 2 ); // Perception
rn -= mod_count * 10; // Other mods
if( rn <= 8 ) {
const std::string startdurability = mod.durability_indicator( true );
const auto destroyed = mod.inc_damage();
const std::string resultdurability = mod.durability_indicator( true );
p->add_msg_if_player( m_bad, _( "You damage your %s trying to modify it! ( %s-> %s)" ),
mod.tname( 1, false ), startdurability, resultdurability );
if( destroyed ) {
p->add_msg_if_player( m_bad, _( "You destroy it!" ) );
p->i_rem_keep_contents( pos );
}
return thread_needed / 2;
} else if( rn <= 10 ) {
p->add_msg_if_player( m_bad,
_( "You fail to modify the clothing, and you waste thread and materials." ) );
p->consume_items( comps, 1, is_crafting_component );
return thread_needed;
} else if( rn <= 14 ) {
p->add_msg_if_player( m_mixed, _( "You modify your %s, but waste a lot of thread." ),
mod.tname() );
p->consume_items( comps, 1, is_crafting_component );
mod.item_tags.insert( the_mod );
return thread_needed;
}
p->add_msg_if_player( m_good, _( "You modify your %s!" ), mod.tname() );
mod.item_tags.insert( the_mod );
p->consume_items( comps, 1, is_crafting_component );
return thread_needed / 2;
}
int iuse::radio_mod( player *p, item *, bool, const tripoint & )
{
if( p->is_npc() ) {
// Now THAT would be kinda cruel
return 0;
}
int inventory_index = g->inv_for_filter( _( "Modify what?" ), []( const item & itm ) {
return itm.has_flag( "RADIO_MODABLE" );
} );
item &modded = p->i_at( inventory_index );
if( modded.is_null() ) {
p->add_msg_if_player( _( "You do not have that item!" ) );
return 0;
}
int choice = uilist( _( "Which signal should activate the item?:" ), {
_( "\"Red\"" ), _( "\"Blue\"" ), _( "\"Green\"" )
} );
std::string newtag;
std::string colorname;
switch( choice ) {
case 0:
newtag = "RADIOSIGNAL_1";
colorname = _( "\"Red\"" );
break;
case 1:
newtag = "RADIOSIGNAL_2";
colorname = _( "\"Blue\"" );
break;
case 2:
newtag = "RADIOSIGNAL_3";
colorname = _( "\"Green\"" );
break;
default:
return 0;
}
if( modded.has_flag( "RADIO_MOD" ) && modded.has_flag( newtag ) ) {
p->add_msg_if_player( _( "This item has been modified this way already." ) );
return 0;
}
remove_radio_mod( modded, *p );
p->add_msg_if_player(
_( "You modify your %1$s to listen for %2$s activation signal on the radio." ),
modded.tname(), colorname );
modded.item_tags.insert( "RADIO_ACTIVATION" );
modded.item_tags.insert( "RADIOCARITEM" );
modded.item_tags.insert( "RADIO_MOD" );
modded.item_tags.insert( newtag );
return 1;
}
int iuse::remove_all_mods( player *p, item *, bool, const tripoint & )
{
if( !p ) {
return 0;
}
auto loc = g->inv_map_splice( []( const item & e ) {
return !e.toolmods().empty();
},
_( "Remove mods from tool?" ), 1,
_( "You don't have any modified tools." ) );
if( !loc ) {
add_msg( m_info, _( "Never mind." ) );
return 0;
}
if( !loc->ammo_remaining() || g->unload( *loc ) ) {
auto mod = std::find_if( loc->contents.begin(), loc->contents.end(), []( const item & e ) {
return e.is_toolmod();
} );
p->i_add_or_drop( *mod );
loc->contents.erase( mod );
remove_radio_mod( *loc, *p );
}
return 0;
}
static bool good_fishing_spot( tripoint pos )
{
std::unordered_set<tripoint> fishable_locations = g->get_fishable_locations( 60, pos );
std::vector<monster *> fishables = g->get_fishable_monsters( fishable_locations );
// isolated little body of water with no definite fish population
oter_id &cur_omt = overmap_buffer.ter( ms_to_omt_copy( g->m.getabs( pos ) ) );
std::string om_id = cur_omt.id().c_str();
if( fishables.empty() && !g->m.has_flag( "CURRENT", pos ) &&
om_id.find( "river_" ) == std::string::npos && !cur_omt->is_lake() && !cur_omt->is_lake_shore() ) {
g->u.add_msg_if_player( m_info, _( "You doubt you will have much luck catching fish here" ) );
return false;
}
return true;
}
int iuse::fishing_rod( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// Long actions - NPCs don't like those yet
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Fish where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( !g->m.has_flag( "FISHABLE", pnt ) ) {
p->add_msg_if_player( m_info, _( "You can't fish there!" ) );
return 0;
}
if( !good_fishing_spot( pnt ) ) {
return 0;
}
p->add_msg_if_player( _( "You cast your line and wait to hook something..." ) );
p->assign_activity( activity_id( "ACT_FISH" ), to_moves<int>( 5_hours ), 0,
p->get_item_position( it ), it->tname() );
p->activity.coord_set = g->get_fishable_locations( 60, pnt );
return 0;
}
int iuse::fish_trap( player *p, item *it, bool t, const tripoint &pos )
{
if( !t ) {
// Handle deploying fish trap.
if( it->active ) {
it->active = false;
return 0;
}
if( it->charges < 0 ) {
it->charges = 0;
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( it->charges == 0 ) {
p->add_msg_if_player( _( "Fish are not foolish enough to go in here without bait." ) );
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Put fish trap where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( !g->m.has_flag( "FISHABLE", pnt ) ) {
p->add_msg_if_player( m_info, _( "You can't fish there!" ) );
return 0;
}
if( !good_fishing_spot( pnt ) ) {
return 0;
}
it->active = true;
it->set_age( 0_turns );
g->m.add_item_or_charges( pnt, *it );
p->i_rem( it );
p->add_msg_if_player( m_info,
_( "You place the fish trap, in three hours or so you may catch some fish." ) );
return 0;
} else {
// Handle processing fish trap over time.
if( it->charges == 0 ) {
it->active = false;
return 0;
}
if( it->age() > 3_hours ) {
it->active = false;
if( !g->m.has_flag( "FISHABLE", pos ) ) {
return 0;
}
int success = -50;
const int surv = p->get_skill_level( skill_survival );
const int attempts = rng( it->charges, it->charges * it->charges );
for( int i = 0; i < attempts; i++ ) {
/** @EFFECT_SURVIVAL randomly increases number of fish caught in fishing trap */
success += rng( surv, surv * surv );
}
it->charges = rng( -1, it->charges );
if( it->charges < 0 ) {
it->charges = 0;
}
int fishes = 0;
if( success < 0 ) {
fishes = 0;
} else if( success < 300 ) {
fishes = 1;
} else if( success < 1500 ) {
fishes = 2;
} else {
fishes = rng( 3, 5 );
}
if( fishes == 0 ) {
it->charges = 0;
p->practice( skill_survival, rng( 5, 15 ) );
return 0;
}
//get the fishables around the trap's spot
std::unordered_set<tripoint> fishable_locations = g->get_fishable_locations( 60, pos );
std::vector<monster *> fishables = g->get_fishable_monsters( fishable_locations );
for( int i = 0; i < fishes; i++ ) {
p->practice( skill_survival, rng( 3, 10 ) );
if( fishables.size() >= 1 ) {
monster *chosen_fish = random_entry( fishables );
// reduce the abstract fish_population marker of that fish
chosen_fish->fish_population -= 1;
if( chosen_fish->fish_population <= 0 ) {
g->catch_a_monster( chosen_fish, pos, p, 300_hours ); //catch the fish!
} else {
g->m.add_item_or_charges( p->pos(), item::make_corpse( chosen_fish->type->id,
calendar::turn + rng( 0_turns,
3_hours ) ) );
}
} else {
//there will always be a chance that the player will get lucky and catch a fish
//not existing in the fishables vector. (maybe it was in range, but wandered off)
//lets say it is a 5% chance per fish to catch
if( one_in( 20 ) ) {
const std::vector<mtype_id> fish_group = MonsterGroupManager::GetMonstersFromGroup(
mongroup_id( "GROUP_FISH" ) );
const mtype_id &fish_mon = random_entry_ref( fish_group );
//Yes, we can put fishes in the trap like knives in the boot,
//and then get fishes via activation of the item,
//but it's not as comfortable as if you just put fishes in the same tile with the trap.
//Also: corpses and comestibles do not rot in containers like this, but on the ground they will rot.
//we don't know when it was caught so use a random turn
g->m.add_item_or_charges( pos, item::make_corpse( fish_mon, it->birthday() + rng( 0_turns,
3_hours ) ) );
break; //this can happen only once
}
}
}
}
return 0;
}
}
int iuse::extinguisher( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
return 0;
}
g->draw();
// If anyone other than the player wants to use one of these,
// they're going to need to figure out how to aim it.
const cata::optional<tripoint> dest_ = choose_adjacent( _( "Spray where?" ) );
if( !dest_ ) {
return 0;
}
tripoint dest = *dest_;
p->moves -= to_moves<int>( 2_seconds );
// Reduce the strength of fire (if any) in the target tile.
g->m.mod_field_intensity( dest, fd_fire, 0 - rng( 2, 3 ) );
// Also spray monsters in that tile.
if( monster *const mon_ptr = g->critter_at<monster>( dest, true ) ) {
monster &critter = *mon_ptr;
critter.moves -= to_moves<int>( 2_seconds );
bool blind = false;
if( one_in( 2 ) && critter.has_flag( MF_SEES ) ) {
blind = true;
critter.add_effect( effect_blind, rng( 1_minutes, 2_minutes ) );
}
if( g->u.sees( critter ) ) {
p->add_msg_if_player( _( "The %s is sprayed!" ), critter.name() );
if( blind ) {
p->add_msg_if_player( _( "The %s looks blinded." ), critter.name() );
}
}
if( critter.made_of( LIQUID ) ) {
if( g->u.sees( critter ) ) {
p->add_msg_if_player( _( "The %s is frozen!" ), critter.name() );
}
critter.apply_damage( p, bp_torso, rng( 20, 60 ) );
critter.set_speed_base( critter.get_speed_base() / 2 );
}
}
// Slightly reduce the strength of fire immediately behind the target tile.
if( g->m.passable( dest ) ) {
dest.x += ( dest.x - p->posx() );
dest.y += ( dest.y - p->posy() );
g->m.mod_field_intensity( dest, fd_fire, std::min( 0 - rng( 0, 1 ) + rng( 0, 1 ), 0 ) );
}
return it->type->charges_to_use();
}
int iuse::rm13armor_off( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
p->add_msg_if_player( m_info, _( "The RM13 combat armor's fuel cells are dead." ) );
return 0;
} else {
std::string oname = it->typeId() + "_on";
p->add_msg_if_player( _( "You activate your RM13 combat armor." ) );
p->add_msg_if_player( _( "Rivtech Model 13 RivOS v2.19: ONLINE." ) );
p->add_msg_if_player( _( "CBRN defense system: ONLINE." ) );
p->add_msg_if_player( _( "Acoustic dampening system: ONLINE." ) );
p->add_msg_if_player( _( "Thermal regulation system: ONLINE." ) );
p->add_msg_if_player( _( "Vision enhancement system: ONLINE." ) );
p->add_msg_if_player( _( "Electro-reactive armor system: ONLINE." ) );
p->add_msg_if_player( _( "All systems nominal." ) );
it->convert( oname ).active = true;
return it->type->charges_to_use();
}
}
int iuse::rm13armor_on( player *p, item *it, bool t, const tripoint & )
{
if( t ) { // Normal use
} else { // Turning it off
std::string oname = it->typeId();
if( oname.length() > 3 && oname.compare( oname.length() - 3, 3, "_on" ) == 0 ) {
oname.erase( oname.length() - 3, 3 );
} else {
debugmsg( "no item type to turn it into (%s)!", oname );
return 0;
}
p->add_msg_if_player( _( "RivOS v2.19 shutdown sequence initiated." ) );
p->add_msg_if_player( _( "Shutting down." ) );
p->add_msg_if_player( _( "Your RM13 combat armor turns off." ) );
it->convert( oname ).active = false;
}
return it->type->charges_to_use();
}
int iuse::unpack_item( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
std::string oname = it->typeId() + "_on";
p->moves -= to_moves<int>( 10_seconds );
p->add_msg_if_player( _( "You unpack your %s for use." ), it->tname() );
it->convert( oname ).active = false;
return 0;
}
int iuse::pack_item( player *p, item *it, bool t, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( t ) { // Normal use
// Numbers below -1 are reserved for worn items
} else if( p->get_item_position( it ) < -1 ) {
p->add_msg_if_player( m_info, _( "You can't pack your %s until you take it off." ),
it->tname() );
return 0;
} else { // Turning it off
std::string oname = it->typeId();
if( oname.length() > 3 && oname.compare( oname.length() - 3, 3, "_on" ) == 0 ) {
oname.erase( oname.length() - 3, 3 );
} else {
debugmsg( "no item type to turn it into (%s)!", oname );
return 0;
}
p->moves -= to_moves<int>( 10_seconds );
p->add_msg_if_player( _( "You pack your %s for storage." ), it->tname() );
it->convert( oname ).active = false;
}
return 0;
}
static int cauterize_elec( player &p, item &it )
{
if( it.charges == 0 && it.ammo_capacity() ) {
p.add_msg_if_player( m_info, _( "You need batteries to cauterize wounds." ) );
return 0;
} else if( !p.has_effect( effect_bite ) && !p.has_effect( effect_bleed ) && !p.is_underwater() ) {
if( ( p.has_trait( trait_MASOCHIST ) || p.has_trait( trait_MASOCHIST_MED ) ||
p.has_trait( trait_CENOBITE ) ) &&
p.query_yn( _( "Cauterize yourself for fun?" ) ) ) {
return cauterize_actor::cauterize_effect( p, it, true ) ? it.type->charges_to_use() : 0;
} else {
p.add_msg_if_player( m_info,
_( "You are not bleeding or bitten, there is no need to cauterize yourself." ) );
return 0;
}
} else if( p.is_npc() || query_yn( _( "Cauterize any open wounds?" ) ) ) {
return cauterize_actor::cauterize_effect( p, it, true ) ? it.type->charges_to_use() : 0;
}
return 0;
}
int iuse::water_purifier( player *p, item *it, bool, const tripoint & )
{
auto obj = g->inv_map_splice( []( const item & e ) {
return !e.contents.empty() && e.contents.front().typeId() == "water";
}, _( "Purify what?" ), 1, _( "You don't have water to purify." ) );
if( !obj ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return 0;
}
item &liquid = obj->contents.front();
if( !it->units_sufficient( *p, liquid.charges ) ) {
p->add_msg_if_player( m_info, _( "That volume of water is too large to purify." ) );
return 0;
}
p->moves -= to_moves<int>( 2_seconds );
liquid.convert( "water_clean" ).poison = 0;
return liquid.charges;
}
int iuse::radio_off( player *p, item *it, bool, const tripoint & )
{
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( _( "It's dead." ) );
} else {
p->add_msg_if_player( _( "You turn the radio on." ) );
it->convert( "radio_on" ).active = true;
}
return it->type->charges_to_use();
}
int iuse::directional_antenna( player *p, item *it, bool, const tripoint & )
{
// Find out if we have an active radio
auto radios = p->items_with( []( const item & it ) {
return it.typeId() == "radio_on";
} );
if( radios.empty() ) {
add_msg( m_info, _( "Must have an active radio to check for signal direction." ) );
return 0;
}
const item radio = *radios.front();
// Find the radio station its tuned to (if any)
const auto tref = overmap_buffer.find_radio_station( radio.frequency );
if( !tref ) {
add_msg( m_info, _( "You can't find the direction if your radio isn't tuned." ) );
return 0;
}
// Report direction.
const auto player_pos = p->global_sm_location();
direction angle = direction_from( player_pos.x, player_pos.y,
tref.abs_sm_pos.x, tref.abs_sm_pos.y );
add_msg( _( "The signal seems strongest to the %s." ), direction_name( angle ) );
return it->type->charges_to_use();
}
int iuse::radio_on( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) {
// Normal use
std::string message = _( "Radio: Kssssssssssssh." );
const auto tref = overmap_buffer.find_radio_station( it->frequency );
if( tref ) {
const auto selected_tower = tref.tower;
if( selected_tower->type == MESSAGE_BROADCAST ) {
message = selected_tower->message;
} else if( selected_tower->type == WEATHER_RADIO ) {
message = weather_forecast( tref.abs_sm_pos );
}
message = obscure_message( message, [&]()->int {
int signal_roll = dice( 10, tref.signal_strength * 3 );
int static_roll = dice( 10, 100 );
if( static_roll > signal_roll )
{
if( static_roll < signal_roll * 1.1 && one_in( 4 ) ) {
return 0;
} else {
return '#';
}
} else
{
return -1;
}
} );
std::vector<std::string> segments = foldstring( message, RADIO_PER_TURN );
int index = calendar::turn % segments.size();
std::stringstream messtream;
messtream << string_format( _( "radio: %s" ), segments[index] );
message = messtream.str();
}
sounds::ambient_sound( pos, 6, sounds::sound_t::speech, message );
if( one_in( 10 ) ) {
sfx::play_variant_sound( "radio", "static", 100, 21, 100 );
sfx::fade_audio_channel( 21, 100 );
} else if( one_in( 10 ) ) {
sfx::play_ambient_variant_sound( "radio", "inaudible_chatter", 100, 21, 100 );
sfx::fade_audio_channel( 21, 100 );
}
} else { // Activated
int ch = 1;
if( it->ammo_remaining() > 0 ) {
ch = uilist( _( "Radio:" ), {
_( "Scan" ), _( "Turn off" )
} );
}
switch( ch ) {
case 0: {
const int old_frequency = it->frequency;
const radio_tower *lowest_tower = nullptr;
const radio_tower *lowest_larger_tower = nullptr;
for( auto &tref : overmap_buffer.find_all_radio_stations() ) {
const auto new_frequency = tref.tower->frequency;
if( new_frequency == old_frequency ) {
continue;
}
if( new_frequency > old_frequency &&
( lowest_larger_tower == nullptr || new_frequency < lowest_larger_tower->frequency ) ) {
lowest_larger_tower = tref.tower;
} else if( lowest_tower == nullptr || new_frequency < lowest_tower->frequency ) {
lowest_tower = tref.tower;
}
}
if( lowest_larger_tower != nullptr ) {
it->frequency = lowest_larger_tower->frequency;
} else if( lowest_tower != nullptr ) {
it->frequency = lowest_tower->frequency;
}
}
break;
case 1:
p->add_msg_if_player( _( "The radio dies." ) );
it->convert( "radio" ).active = false;
break;
default:
break;
}
}
return it->type->charges_to_use();
}
int iuse::noise_emitter_off( player *p, item *it, bool, const tripoint & )
{
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( _( "It's dead." ) );
} else {
p->add_msg_if_player( _( "You turn the noise emitter on." ) );
it->convert( "noise_emitter_on" ).active = true;
}
return it->type->charges_to_use();
}
int iuse::noise_emitter_on( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) { // Normal use
//~ the sound of a noise emitter when turned on
sounds::sound( pos, 30, sounds::sound_t::alarm, _( "KXSHHHHRRCRKLKKK!" ), true, "tool",
"noise_emitter" );
} else { // Turning it off
p->add_msg_if_player( _( "The infernal racket dies as the noise emitter turns off." ) );
it->convert( "noise_emitter" ).active = false;
}
return it->type->charges_to_use();
}
int iuse::ma_manual( player *p, item *it, bool, const tripoint & )
{
// [CR] - should NPCs just be allowed to learn this stuff? Just like that?
const matype_id style_to_learn = martial_art_learned_from( *it->type );
const martialart &ma = style_to_learn.obj();
if( !style_to_learn.is_valid() ) {
// debugmsg will already have been sent by .obj call
return 0;
}
p->ma_styles.push_back( style_to_learn );
p->add_msg_if_player( m_good, _( "You learn the essential elements of %s." ),
_( ma.name ) );
p->add_msg_if_player( m_info, _( "%s to select martial arts style." ),
press_x( ACTION_PICK_STYLE ) );
return 1;
}
static bool pry_nails( player &p, const ter_id &type, const tripoint &pnt )
{
int nails = 0;
int boards = 0;
ter_id newter;
if( type == t_fence ) {
nails = 6;
boards = 3;
newter = t_fence_post;
p.add_msg_if_player( _( "You pry out the fence post." ) );
} else if( type == t_window_boarded ) {
nails = 8;
boards = 4;
newter = t_window_frame;
p.add_msg_if_player( _( "You pry the boards from the window." ) );
} else if( type == t_window_boarded_noglass ) {
nails = 8;
boards = 4;
newter = t_window_empty;
p.add_msg_if_player( _( "You pry the boards from the window frame." ) );
} else if( type == t_door_boarded || type == t_door_boarded_damaged ||
type == t_rdoor_boarded || type == t_rdoor_boarded_damaged ||
type == t_door_boarded_peep || type == t_door_boarded_damaged_peep ) {
nails = 8;
boards = 4;
if( type == t_door_boarded ) {
newter = t_door_c;
} else if( type == t_door_boarded_damaged ) {
newter = t_door_b;
} else if( type == t_door_boarded_peep ) {
newter = t_door_c_peep;
} else if( type == t_door_boarded_damaged_peep ) {
newter = t_door_b_peep;
} else if( type == t_rdoor_boarded ) {
newter = t_rdoor_c;
} else { // if (type == t_rdoor_boarded_damaged)
newter = t_rdoor_b;
}
p.add_msg_if_player( _( "You pry the boards from the door." ) );
} else {
return false;
}
p.practice( skill_fabrication, 1, 1 );
p.moves -= to_moves<int>( 30_seconds );
g->m.spawn_item( p.pos(), "nail", 0, nails );
g->m.spawn_item( p.pos(), "2x4", boards );
g->m.ter_set( pnt, newter );
return true;
}
int iuse::hammer( player *p, item *it, bool, const tripoint & )
{
// If anyone other than the player wants to use one of these,
// they're going to need to figure out how to aim it.
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Pry where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
p->add_msg_if_player( _( "You try to hit yourself with the hammer." ) );
p->add_msg_if_player( _( "But you can't touch this." ) );
return 0;
}
return this->crowbar( p, it, false, pnt );
}
int iuse::crowbar( player *p, item *it, bool, const tripoint &pos )
{
// TODO: Make this 3D now that NPCs get to use items
tripoint pnt = pos;
if( pos == p->pos() ) {
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Pry where?" ) );
if( !pnt_ ) {
return 0;
}
pnt = *pnt_;
} // else it is already set to pos in the line above if
if( pnt == p->pos() ) {
p->add_msg_if_player( m_info, _( "You attempt to pry open your wallet "
"but alas. You are just too miserly." ) );
return 0;
}
ter_id type = g->m.ter( pnt );
const char *succ_action;
const char *fail_action;
ter_id new_type = t_null;
bool noisy;
int pry_quality;
int difficulty;
if( type == t_door_locked || type == t_door_locked_alarm || type == t_door_locked_interior ) {
succ_action = _( "You pry open the door." );
fail_action = _( "You pry, but cannot pry open the door." );
new_type = t_door_o;
pry_quality = 2;
noisy = true;
difficulty = 6;
} else if( type == t_door_locked_peep ) {
succ_action = _( "You pry open the door." );
fail_action = _( "You pry, but cannot pry open the door." );
new_type = t_door_o_peep;
pry_quality = 2;
noisy = true;
difficulty = 6;
} else if( type == t_door_c ) {
p->add_msg_if_player( m_info, _( "You notice the door is unlocked, so you simply open it." ) );
g->m.ter_set( pnt, t_door_o );
p->mod_moves( -100 );
return 0;
} else if( type == t_door_c_peep ) {
p->add_msg_if_player( m_info, _( "You notice the door is unlocked, so you simply open it." ) );
g->m.ter_set( pnt, t_door_o_peep );
p->mod_moves( -100 );
return 0;
} else if( type == t_manhole_cover ) {
succ_action = _( "You lift the manhole cover." );
fail_action = _( "You pry, but cannot lift the manhole cover." );
pry_quality = 1;
new_type = t_manhole;
noisy = false;
difficulty = 4;
} else if( g->m.furn( pnt ) == f_crate_c ) {
succ_action = _( "You pop open the crate." );
fail_action = _( "You pry, but cannot pop open the crate." );
pry_quality = 1;
noisy = true;
difficulty = 6;
} else if( g->m.furn( pnt ) == f_coffin_c ) {
succ_action = _( "You wedge open the coffin." );
fail_action = _( "You pry, but the coffin remains closed." );
pry_quality = 2;
noisy = true;
difficulty = 5;
} else if( type == t_window_domestic || type == t_curtains || type == t_window_no_curtains ) {
succ_action = _( "You pry open the window." );
fail_action = _( "You pry, but cannot pry open the window." );
new_type = ( type == t_window_no_curtains ) ? t_window_no_curtains_open : t_window_open;
pry_quality = 2;
noisy = true;
difficulty = 6;
} else if( pry_nails( *p, type, pnt ) ) {
return it->type->charges_to_use();
} else {
p->add_msg_if_player( m_info, _( "You can't pry that." ) );
return 0;
}
// Doors need PRY 2 which is on a crowbar, crates need PRY 1 which is on a crowbar
// & a claw hammer.
// The iexamine function for crate supplies a hammer object.
// So this stops the player (A)ctivating a Hammer with a Crowbar in their backpack
// then managing to open a door.
const int pry_level = it->get_quality( quality_id( "PRY" ) );
if( pry_level < pry_quality ) {
p->add_msg_if_player( _( "You can't get sufficient leverage to open that with your %s." ),
it->tname() );
p->mod_moves( 10 ); // spend a few moves trying it.
return 0;
}
// For every level of PRY over the requirement, remove n from the difficulty (so -2 with a PRY 4 tool)
difficulty -= ( pry_level - pry_quality );
/** @EFFECT_STR speeds up crowbar prying attempts */
p->mod_moves( -std::max( 20, difficulty * 20 - p->str_cur * 5 ) );
/** @EFFECT_STR increases chance of crowbar prying success */
if( dice( 4, difficulty ) < dice( 4, p->str_cur ) ) {
p->add_msg_if_player( m_good, succ_action );
if( g->m.furn( pnt ) == f_crate_c ) {
g->m.furn_set( pnt, f_crate_o );
} else if( g->m.furn( pnt ) == f_coffin_c ) {
g->m.furn_set( pnt, f_coffin_o );
} else {
g->m.ter_set( pnt, new_type );
}
if( noisy ) {
sounds::sound( pnt, 12, sounds::sound_t::combat, _( "crunch!" ), true, "tool", "crowbar" );
}
if( type == t_manhole_cover ) {
g->m.spawn_item( pnt, "manhole_cover" );
}
if( type == t_door_locked_alarm ) {
p->add_memorial_log( pgettext( "memorial_male", "Set off an alarm." ),
pgettext( "memorial_female", "Set off an alarm." ) );
sounds::sound( p->pos(), 40, sounds::sound_t::alarm, _( "an alarm sound!" ), true, "environment",
"alarm" );
if( !g->events.queued( EVENT_WANTED ) ) {
g->events.add( EVENT_WANTED, calendar::turn + 30_minutes, 0, p->global_sm_location() );
}
}
} else {
if( type == t_window_domestic || type == t_curtains ) {
//chance of breaking the glass if pry attempt fails
/** @EFFECT_STR reduces chance of breaking window with crowbar */
/** @EFFECT_MECHANICS reduces chance of breaking window with crowbar */
if( dice( 4, difficulty ) > dice( 2, p->get_skill_level( skill_mechanics ) ) + dice( 2,
p->str_cur ) ) {
p->add_msg_if_player( m_mixed, _( "You break the glass." ) );
sounds::sound( pnt, 24, sounds::sound_t::combat, _( "glass breaking!" ), true, "smash", "glass" );
g->m.ter_set( pnt, t_window_frame );
g->m.spawn_item( pnt, "sheet", 2 );
g->m.spawn_item( pnt, "stick" );
g->m.spawn_item( pnt, "string_36" );
return it->type->charges_to_use();
}
}
p->add_msg_if_player( fail_action );
}
return it->type->charges_to_use();
}
int iuse::makemound( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Till soil where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
add_msg( m_info, _( "You think about jumping on a shovel, but then change up your mind." ) );
return 0;
}
if( g->m.has_flag( "PLOWABLE", pnt ) && !g->m.has_flag( "PLANT", pnt ) ) {
p->add_msg_if_player( _( "You churn up the earth here." ) );
p->mod_moves( -300 );
g->m.ter_set( pnt, t_dirtmound );
return it->type->charges_to_use();
} else {
p->add_msg_if_player( _( "You can't churn up this ground." ) );
return 0;
}
}
struct digging_moves_and_byproducts {
int moves;
int spawn_count;
std::string byproducts_item_group;
ter_id result_terrain;
};
static digging_moves_and_byproducts dig_pit_moves_and_byproducts( player *p, item *it, bool deep,
bool channel )
{
// When we dig, we're generally digging out either a deep or shallow pit.
//
// The dimensions are a little hand-wavey... based on our exactly
// as-big-as-they-need-to-be tile sizes, we could assume that the width and height
// are 1 meter each. Our pit could be a square as well, or we could assume it's
// circular and 1 meter in diameter.
//
// Depth is even less rigidly defined, both in terms of "what is a z-level", and in
// terms of how the deep and shallow pits are used in game.
//
// A shallow pit gets used to do things like build an improvised shelter or start
// the foundation for a wall, and is ostensibly a relatively quick effort: a
// survior might get a crude digging implement (e.g. digging stick) and attempt to
// dig out the start of an improvised shelter, or they might have a proper shovel
// and be digging a foundation footing. Referencing the 2018 IBC
// https://codes.iccsafe.org/content/IRC2018/chapter-4-foundations we can see in
// R403.1.4 the requirement that "exterior footings shall be placed not less than
// 12 inches (305 mm) below the undisturbed ground surface. You'd need even more
// space for the actual footing, but that's close enough. I don't think surviviors
// care about building to code, but let's call it 12 inches (0.3048 meters) at the
// maximum for a shallow pit depth.
//
// The deep pit is a little more complicated because it gets used for more complex
// constructions like traps, reinforced concrete wall footings, palisades, wells,
// root cellars, and the like. The depth requirements for those are quite varied,
// but let's just throw some number around to get a feel for it: say at least 2
// meters deep to be an effective pit trap. The USGS maintains
// depth-to-ground-water-level records and provides them at
// https://waterdata.usgs.gov/nwis/current/?type=gw. A cursory review (and
// remembering it's seasonal) shows values in Massachusetts ranging from 122 feet
// and 1 foot, and values between 3 to 9 feet aren't uncommon. Root cellars are
// ideally constructed where the ground temperature has stabilized, below the frost
// line. This depth varies by location, but is somewhere in the 1 to 3 meter range.
//
// With all of that in hand, let's make some estimates.
//
// A shallow pit is a circular pit 1 meter in diameter and 0.3048 meters deep,
// which works out to... ~0.239 m^3. That's so close, let's just call it 0.25 m^3
// or 250 liters.
//
// A deep pit is a rectangular pit 1m x 1m x 2m, or 2 m^3, or 2000 liters.
//
// Now, for how long that takes, things are going to get even more subjective and
// couched in assumptions. Let's assume a single individual, digging in optimally
// diggable soil(rather than something requiring a pickaxe to break the soil or
// requiring saws to remove tree roots), using an appropriate (but manual)
// implement designed for the task (e.g. a shovel). The Canadian Centre for
// Occupational Health and Safety has some interesting recommendations on the rate
// of shovelling, weight of the load, and throw distance at
// https://www.ccohs.ca/oshanswers/ergonomics/shovel.html. Of particular interest
// is the table of "recommended workload for continuous shovelling", which gives a
// weight per minute and a total weight per 15 minutes, as well as a description of
// the conditions. It also discusses the need to take breaks, for example
// alternating 15 minutes of shoveling and 15 minutes of rest in extreme
// conditions. Taking all that into consideration, I'm going to call it 10
// scoops/min * 5 kg/scoop for 15 minutes followed by 15 minutes of rest, or
// effectively 25 kg/min.
//
// Now we need to bring the weights and volumes together. Again, more hand waving
// as the soil composition is going to have a big influence on this. The
// engineering toolbox https://www.engineeringtoolbox.com/dirt-mud-densities-d_1727.html
// lists the density of wet and dry versions of many materials. I'm going with the
// assumption that this is moist/wet soil that includes clay, silt, load, as well as
// some rock, so I'll just call it 1700 kg/m^3 on average.
//
// Shallow pit is 0.25 m^3 * 1700 kg/m^3, or 425 kg.
// Deep pit is 2 m^3 * 1700 kg/m^3, or 3400 kg.
//
// We'll do some variables below, but for reference:
// Shallow pit: 425 kg / 25 kg/min = 17 minutes
// Deep pit: 3400 kg / 25 kg/min = 136 minutes
//
// Now, one addendum: we're digging our deep pit in the location that we've already
// dug the shallow pit, so really we should exclude the shallow pit volume from the
// deep when counting the amount of work we need to do.
//
// Adjusted deep pit: ( 3400 kg - 425 kg ) / 25 kg/min = 119 minutes
constexpr double shallow_pit_volume_m3 = 0.25;
constexpr double deep_pit_volume_m3 = 2;
constexpr int dig_rate_kg_min = 25;
constexpr int material_density_kg_m3 = 1700;
// At the time of writing this, a shovel is DIG 3, which is what the numbers are
// balanced around.
constexpr double baseline_dig_quality = 3;
// Get the dig quality of the tool.
const int quality = it->get_quality( DIG );
// Dig quality affects the dig rate linearly relative to baseline dig quality
const double tool_dig_rate = dig_rate_kg_min * quality / baseline_dig_quality;
///\EFFECT_STR modifies dig rate
// Adjust the dig rate by 2 kg/min per point of strength more/less than 10.
const double player_dig_rate = std::max( 1.0, tool_dig_rate + ( p->str_cur - 10 ) * 2 );
// Figure out the volume of the pit we're digging.
// Subtract the shallow volume from the deep, since we already dug that.
const double volume_m3 = deep ? ( deep_pit_volume_m3 - shallow_pit_volume_m3 ) :
shallow_pit_volume_m3;
// And now determine the moves...
int dig_minutes = volume_m3 * material_density_kg_m3 / player_dig_rate;
int moves = to_moves<int>( time_duration::from_minutes( dig_minutes ) );
// Modify the number of moves based on the help.
// TODO: this block of code is all over the place and could probably be consolidated.
const int helpersize = g->u.get_num_crafting_helpers( 3 );
moves = moves * ( 1 - ( helpersize / 10 ) );
ter_id result_terrain;
if( channel ) {
result_terrain = ter_id( "t_water_moving_sh" );
} else {
result_terrain = deep ? ter_id( "t_pit" ) : ter_id( "t_pit_shallow" );
}
return { moves, static_cast<int>( volume_m3 / 0.05 ), "digging_soil_loam_50L", result_terrain };
}
int iuse::dig( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const tripoint dig_point = p->pos();
const bool can_dig_here = g->m.has_flag( "DIGGABLE", dig_point ) &&
!g->m.has_furn( dig_point ) &&
g->m.tr_at( dig_point ).is_null() &&
( g->m.ter( dig_point ) == t_grave_new || g->m.i_at( dig_point ).empty() ) &&
!g->m.veh_at( dig_point );
if( !can_dig_here ) {
p->add_msg_if_player(
_( "You can't dig a pit in this location. Ensure it is clear diggable ground with no items or obstacles." ) );
return 0;
}
const bool can_deepen = g->m.has_flag( "DIGGABLE_CAN_DEEPEN", dig_point );
const bool grave = g->m.ter( dig_point ) == t_grave;
if( !p->crafting_inventory().has_quality( DIG, 2 ) ) {
if( can_deepen ) {
p->add_msg_if_player( _( "You can't deepen this pit without a proper shovel." ) );
return 0;
} else if( grave ) {
p->add_msg_if_player( _( "You can't exhume a grave without a proper shovel." ) );
return 0;
}
}
const std::function<bool( tripoint )> f = []( tripoint p ) {
return g->m.passable( p );
};
const cata::optional<tripoint> pnt_ = choose_adjacent_highlight(
_( "Deposit excavated materials where?" ), f, false );
if( !pnt_ ) {
return 0;
}
const tripoint deposit_point = *pnt_;
if( !g->m.passable( deposit_point ) ) {
p->add_msg_if_player(
_( "You can't deposit the excavated materials onto an impassable location." ) );
return 0;
}
if( grave ) {
if( g->u.has_trait( trait_id( "SPIRITUAL" ) ) && !g->u.has_trait( trait_id( "PSYCHOPATH" ) ) &&
g->u.query_yn( _( "Would you really touch the sacred resting place of the dead?" ) ) ) {
add_msg( m_info, _( "Exhuming a grave is really against your beliefs." ) );
g->u.add_morale( MORALE_GRAVEDIGGER, -50, -100, 48_hours, 12_hours );
if( one_in( 3 ) ) {
g->u.vomit();
}
} else if( g->u.has_trait( trait_id( "PSYCHOPATH" ) ) ) {
p->add_msg_if_player( m_good,
_( "Exhuming a grave is fun now, where there is no one to object." ) );
g->u.add_morale( MORALE_GRAVEDIGGER, 25, 50, 2_hours, 1_hours );
} else if( !g->u.has_trait( trait_id( "EATDEAD" ) ) &&
!g->u.has_trait( trait_id( "SAPROVORE" ) ) ) {
p->add_msg_if_player( m_bad, _( "Exhuming this grave is utterly disgusting!" ) );
g->u.add_morale( MORALE_GRAVEDIGGER, -25, -50, 2_hours, 1_hours );
if( one_in( 5 ) ) {
p->vomit();
}
}
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
digging_moves_and_byproducts moves_and_byproducts = dig_pit_moves_and_byproducts( p, it,
can_deepen, false );
player_activity act( activity_id( "ACT_DIG" ), moves_and_byproducts.moves, -1,
p->get_item_position( it ) );
act.placement = dig_point;
act.values.emplace_back( moves_and_byproducts.spawn_count );
act.str_values.emplace_back( moves_and_byproducts.byproducts_item_group );
act.str_values.emplace_back( moves_and_byproducts.result_terrain.id().str() );
act.coords.emplace_back( deposit_point );
p->assign_activity( act );
return it->type->charges_to_use();
}
int iuse::dig_channel( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const tripoint dig_point = p->pos();
tripoint north = dig_point + point( 0, -1 );
tripoint south = dig_point + point( 0, 1 );
tripoint west = dig_point + point( -1, 0 );
tripoint east = dig_point + point( 1, 0 );
const bool can_dig_here = g->m.has_flag( "DIGGABLE", dig_point ) && !g->m.has_furn( dig_point ) &&
g->m.tr_at( dig_point ).is_null() && g->m.i_at( dig_point ).empty() && !g->m.veh_at( dig_point ) &&
( g->m.has_flag( "CURRENT", north ) || g->m.has_flag( "CURRENT", south ) ||
g->m.has_flag( "CURRENT", east ) || g->m.has_flag( "CURRENT", west ) );
if( !can_dig_here ) {
p->add_msg_if_player(
_( "You can't dig a channel in this location. Ensure it is clear diggable ground with no items or obstacles, adjacent to flowing water." ) );
return 0;
}
const std::function<bool( tripoint )> f = []( tripoint p ) {
return g->m.passable( p );
};
const cata::optional<tripoint> pnt_ = choose_adjacent_highlight(
_( "Deposit excavated materials where?" ), f, false );
if( !pnt_ ) {
return 0;
}
const tripoint deposit_point = *pnt_;
if( !g->m.passable( deposit_point ) ) {
p->add_msg_if_player(
_( "You can't deposit the excavated materials onto an impassable location." ) );
return 0;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
digging_moves_and_byproducts moves_and_byproducts = dig_pit_moves_and_byproducts( p, it, false,
true );
player_activity act( activity_id( "ACT_DIG_CHANNEL" ), moves_and_byproducts.moves, -1,
p->get_item_position( it ) );
act.placement = dig_point;
act.values.emplace_back( moves_and_byproducts.spawn_count );
act.str_values.emplace_back( moves_and_byproducts.byproducts_item_group );
act.str_values.emplace_back( moves_and_byproducts.result_terrain.id().str() );
act.coords.emplace_back( deposit_point );
p->assign_activity( act );
return it->type->charges_to_use();
}
int iuse::fill_pit( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Fill which pit or mound?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
add_msg( m_info, _( "You decide not to bury yourself that early." ) );
return 0;
}
int moves;
if( g->m.ter( pnt ) == t_pit || g->m.ter( pnt ) == t_pit_spiked ||
g->m.ter( pnt ) == t_pit_glass || g->m.ter( pnt ) == t_pit_corpsed ) {
moves = to_turns<int>( time_duration::from_minutes( 15 ) );
} else if( g->m.ter( pnt ) == t_pit_shallow ) {
moves = to_turns<int>( time_duration::from_minutes( 10 ) );
} else if( g->m.ter( pnt ) == t_dirtmound ) {
moves = to_turns<int>( time_duration::from_minutes( 5 ) );
} else {
p->add_msg_if_player( _( "There is nothing to fill." ) );
return 0;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
const int helpersize = g->u.get_num_crafting_helpers( 3 );
moves = moves * ( 1 - ( helpersize / 10 ) );
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
p->assign_activity( activity_id( "ACT_FILL_PIT" ), moves, -1, p->get_item_position( it ) );
p->activity.placement = pnt;
return it->type->charges_to_use();
}
/**
* Explanation of ACT_CLEAR_RUBBLE activity values:
*
* coords[0]: Where the rubble is.
* index: The bonus, for calculating hunger and thirst penalties.
*/
int iuse::clear_rubble( player *p, item *it, bool, const tripoint & )
{
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Clear rubble where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( g->m.has_flag( "RUBBLE", pnt ) ) {
int bonus = std::max( it->get_quality( quality_id( "DIG" ) ) - 1, 1 );
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
const int helpersize = g->u.get_num_crafting_helpers( 3 );
const int moves = to_moves<int>( 30_seconds ) * ( 1 - ( helpersize / 10 ) );
player_activity act( activity_id( "ACT_CLEAR_RUBBLE" ), moves / bonus, bonus );
p->assign_activity( act );
p->activity.placement = pnt;
return it->type->charges_to_use();
} else {
p->add_msg_if_player( m_bad, _( "There's no rubble to clear." ) );
return 0;
}
}
void act_vehicle_siphon( vehicle * ); // veh_interact.cpp
int iuse::siphon( player *p, item *it, bool, const tripoint & )
{
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Siphon from where?" ) );
if( !pnt_ ) {
return 0;
}
const optional_vpart_position vp = g->m.veh_at( *pnt_ );
if( !vp ) {
p->add_msg_if_player( m_info, _( "There's no vehicle there." ) );
return 0;
}
act_vehicle_siphon( &vp->vehicle() );
return it->type->charges_to_use();
}
static int toolweapon_off( player &p, item &it, const bool fast_startup,
const bool condition, const int volume,
const std::string &msg_success, const std::string &msg_failure )
{
p.moves -= fast_startup ? 60 : 80;
if( condition && it.ammo_remaining() > 0 ) {
if( it.typeId() == "chainsaw_off" ) {
sfx::play_variant_sound( "chainsaw_cord", "chainsaw_on", sfx::get_heard_volume( p.pos() ) );
sfx::play_variant_sound( "chainsaw_start", "chainsaw_on", sfx::get_heard_volume( p.pos() ) );
sfx::play_ambient_variant_sound( "chainsaw_idle", "chainsaw_on", sfx::get_heard_volume( p.pos() ),
18, 1000 );
sfx::play_ambient_variant_sound( "weapon_theme", "chainsaw", sfx::get_heard_volume( p.pos() ), 19,
3000 );
}
sounds::sound( p.pos(), volume, sounds::sound_t::combat, msg_success );
it.convert( it.typeId().substr( 0, it.typeId().size() - 4 ) + "_on" ); // 4 is the length of "_off".
it.active = true;
} else {
if( it.typeId() == "chainsaw_off" ) {
sfx::play_variant_sound( "chainsaw_cord", "chainsaw_on", sfx::get_heard_volume( p.pos() ) );
}
p.add_msg_if_player( msg_failure );
}
return it.type->charges_to_use();
}
int iuse::combatsaw_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
true,
!p->is_underwater(),
30, _( "With a snarl, the combat chainsaw screams to life!" ),
_( "You yank the cord, but nothing happens." ) );
}
int iuse::e_combatsaw_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
true,
!p->is_underwater(),
30, _( "With a snarl, the electric combat chainsaw screams to life!" ),
_( "You flip the switch, but nothing happens." ) );
}
int iuse::chainsaw_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
false,
rng( 0, 10 ) - it->damage_level( 4 ) > 5 && !p->is_underwater(),
20, _( "With a roar, the chainsaw leaps to life!" ),
_( "You yank the cord, but nothing happens." ) );
}
int iuse::elec_chainsaw_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
false,
rng( 0, 10 ) - it->damage_level( 4 ) > 5 && !p->is_underwater(),
20, _( "With a roar, the electric chainsaw leaps to life!" ),
_( "You flip the switch, but nothing happens." ) );
}
int iuse::cs_lajatang_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
false,
rng( 0, 10 ) - it->damage_level( 4 ) > 5 && it->ammo_remaining() > 1 && !p->is_underwater(),
40, _( "With a roar, the chainsaws leap to life!" ),
_( "You yank the cords, but nothing happens." ) );
}
int iuse::ecs_lajatang_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
false,
rng( 0, 10 ) - it->damage_level( 4 ) > 5 && it->ammo_remaining() > 1 && !p->is_underwater(),
40, _( "With a buzz, the chainsaws leap to life!" ),
_( "You flip the on switch, but nothing happens." ) );
}
int iuse::carver_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
false,
true,
20, _( "The electric carver's serrated blades start buzzing!" ),
_( "You pull the trigger, but nothing happens." ) );
}
int iuse::trimmer_off( player *p, item *it, bool, const tripoint & )
{
return toolweapon_off( *p, *it,
false,
rng( 0, 10 ) - it->damage_level( 4 ) > 3,
15, _( "With a roar, the hedge trimmer leaps to life!" ),
_( "You yank the cord, but nothing happens." ) );
}
static int toolweapon_on( player &p, item &it, const bool t,
const std::string &tname, const bool works_underwater,
const int sound_chance, const int volume,
const std::string &sound, const bool double_charge_cost = false )
{
std::string off_type =
it.typeId().substr( 0, it.typeId().size() - 3 ) +
// 3 is the length of "_on".
"_off";
if( t ) { // Effects while simply on
if( double_charge_cost && it.ammo_remaining() > 0 ) {
it.ammo_consume( 1, p.pos() );
}
if( !works_underwater && p.is_underwater() ) {
p.add_msg_if_player( _( "Your %s gurgles in the water and stops." ), tname );
it.convert( off_type ).active = false;
} else if( one_in( sound_chance ) ) {
sounds::ambient_sound( p.pos(), volume, sounds::sound_t::activity, sound );
}
} else { // Toggling
if( it.typeId() == "chainsaw_on" ) {
sfx::play_variant_sound( "chainsaw_stop", "chainsaw_on", sfx::get_heard_volume( p.pos() ) );
sfx::fade_audio_channel( 18, 100 );
sfx::fade_audio_channel( 19, 3000 );
}
p.add_msg_if_player( _( "Your %s goes quiet." ), tname );
it.convert( off_type ).active = false;
}
return it.type->charges_to_use();
}
int iuse::combatsaw_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "combat chainsaw" ),
false,
12, 18, _( "Your combat chainsaw growls." ) );
}
int iuse::e_combatsaw_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "electric combat chainsaw" ),
false,
12, 18, _( "Your electric combat chainsaw growls." ) );
}
int iuse::chainsaw_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "chainsaw" ),
false,
15, 12, _( "Your chainsaw rumbles." ) );
}
int iuse::elec_chainsaw_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "electric chainsaw" ),
false,
15, 12, _( "Your electric chainsaw rumbles." ) );
}
int iuse::cs_lajatang_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "chainsaw lajatang" ),
false,
15, 12, _( "Your chainsaws rumble." ),
true );
// The chainsaw lajatang drains 2 charges per turn, since
// there are two chainsaws.
}
int iuse::ecs_lajatang_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "electric chainsaw lajatang" ),
false,
15, 12, _( "Your chainsaws buzz." ),
true );
// The chainsaw lajatang drains 2 charges per turn, since
// there are two chainsaws.
}
int iuse::carver_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "electric carver" ),
true,
10, 8, _( "Your electric carver buzzes." ) );
}
int iuse::trimmer_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "hedge trimmer" ),
true,
15, 10, _( "Your hedge trimmer rumbles." ) );
}
int iuse::circsaw_on( player *p, item *it, bool t, const tripoint & )
{
return toolweapon_on( *p, *it, t, _( "circular saw" ),
true,
15, 7, _( "Your circular saw buzzes." ) );
}
int iuse::jackhammer( player *p, item *it, bool, const tripoint &pos )
{
// use has_enough_charges to check for UPS availability
// p is assumed to exist for iuse cases
if( !p->has_enough_charges( *it, false ) ) {
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
tripoint pnt = pos;
if( pos == p->pos() ) {
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Drill where?" ) );
if( !pnt_ ) {
return 0;
}
pnt = *pnt_;
}
if( !g->m.has_flag( "MINEABLE", pnt ) ) {
p->add_msg_if_player( m_info, _( "You can't drill there." ) );
return 0;
}
if( g->m.veh_at( pnt ) ) {
p->add_msg_if_player( _( "There's a vehicle in the way!" ) );
return 0;
}
int turns = to_moves<int>( 30_minutes );
if( g->m.move_cost( pnt ) == 2 ) {
// We're breaking up some flat surface like pavement, which is much easier
turns /= 2;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
const int helpersize = g->u.get_num_crafting_helpers( 3 );
turns *= ( 1 - ( helpersize / 10 ) );
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
p->assign_activity( activity_id( "ACT_JACKHAMMER" ), turns, -1, p->get_item_position( it ) );
p->activity.placement = pnt;
p->add_msg_if_player( _( "You start drilling into the %1$s with your %2$s." ),
g->m.tername( pnt ), it->tname() );
return it->type->charges_to_use();
}
int iuse::pickaxe( player *p, item *it, bool, const tripoint &pos )
{
if( p->is_npc() ) {
// Long action
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
tripoint pnt = pos;
if( pos == p->pos() ) {
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Mine where?" ) );
if( !pnt_ ) {
return 0;
}
pnt = *pnt_;
}
if( !g->m.has_flag( "MINEABLE", pnt ) ) {
p->add_msg_if_player( m_info, _( "You can't mine there." ) );
return 0;
}
if( g->m.veh_at( pnt ) ) {
p->add_msg_if_player( _( "There's a vehicle in the way!" ) );
return 0;
}
int moves = to_moves<int>( 20_minutes );
moves += ( ( MAX_STAT + 4 ) - std::min( p->str_cur, MAX_STAT ) ) * to_moves<int>( 5_minutes );
if( g->m.move_cost( pnt ) == 2 ) {
// We're breaking up some flat surface like pavement, which is much easier
moves /= 2;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
const int helpersize = g->u.get_num_crafting_helpers( 3 );
moves *= ( 1 - ( helpersize / 10 ) );
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
p->assign_activity( activity_id( "ACT_PICKAXE" ), moves, -1, p->get_item_position( it ) );
p->activity.placement = pnt;
p->add_msg_if_player( _( "You strike the %1$s with your %2$s." ),
g->m.tername( pnt ), it->tname() );
return 0; // handled when the activity finishes
}
int iuse::burrow( player *p, item *it, bool, const tripoint &pos )
{
if( p->is_npc() ) {
// Long action
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
tripoint pnt = pos;
if( pos == p->pos() ) {
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Burrow where?" ) );
if( !pnt_ ) {
return 0;
}
pnt = *pnt_;
}
if( !g->m.has_flag( "MINEABLE", pnt ) ) {
p->add_msg_if_player( m_info, _( "You can't burrow there." ) );
return 0;
}
if( g->m.veh_at( pnt ) ) {
p->add_msg_if_player( _( "There's a vehicle in the way!" ) );
return 0;
}
int moves = to_moves<int>( 5_minutes );
moves += ( ( MAX_STAT + 3 ) - std::min( p->str_cur, MAX_STAT ) ) * to_moves<int>( 2_minutes );
if( g->m.move_cost( pnt ) == 2 ) {
// We're breaking up some flat surface like pavement, which is much easier
moves /= 2;
}
p->assign_activity( activity_id( "ACT_BURROW" ), moves, -1, 0 );
p->activity.placement = pnt;
p->add_msg_if_player( _( "You start tearing into the %1$s with your %2$s." ),
g->m.tername( pnt ), it->tname() );
return 0; // handled when the activity finishes
}
int iuse::geiger( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) { // Every-turn use when it's on
const int rads = g->m.get_radiation( pos );
if( rads == 0 ) {
return it->type->charges_to_use();
}
std::string description = rads > 50 ? _( "buzzing" ) :
rads > 25 ? _( "rapid clicking" ) : _( "clicking" );
std::string sound_var = rads > 50 ? _( "geiger_high" ) :
rads > 25 ? _( "geiger_medium" ) : _( "geiger_low" );
sounds::sound( pos, 6, sounds::sound_t::alarm, description, true, "tool", sound_var );
if( !p->can_hear( pos, 6 ) ) {
// can not hear it, but may have alarmed other creatures
return it->type->charges_to_use();
}
if( rads > 50 ) {
add_msg( m_warning, _( "The geiger counter buzzes intensely." ) );
} else if( rads > 35 ) {
add_msg( m_warning, _( "The geiger counter clicks wildly." ) );
} else if( rads > 25 ) {
add_msg( m_warning, _( "The geiger counter clicks rapidly." ) );
} else if( rads > 15 ) {
add_msg( m_warning, _( "The geiger counter clicks steadily." ) );
} else if( rads > 8 ) {
add_msg( m_warning, _( "The geiger counter clicks slowly." ) );
} else if( rads > 4 ) {
add_msg( _( "The geiger counter clicks intermittently." ) );
} else {
add_msg( _( "The geiger counter clicks once." ) );
}
return it->type->charges_to_use();
}
// Otherwise, we're activating the geiger counter
if( it->typeId() == "geiger_on" ) {
add_msg( _( "The geiger counter's SCANNING LED turns off." ) );
it->convert( "geiger_off" ).active = false;
return 0;
}
int ch = uilist( _( "Geiger counter:" ), {
_( "Scan yourself" ), _( "Scan the ground" ), _( "Turn continuous scan on" )
} );
switch( ch ) {
case 0:
p->add_msg_if_player( m_info, _( "Your radiation level: %d mSv (%d mSv from items)" ), p->radiation,
p->leak_level( "RADIOACTIVE" ) );
break;
case 1:
p->add_msg_if_player( m_info, _( "The ground's radiation level: %d mSv/h" ),
g->m.get_radiation( p->pos() ) );
break;
case 2:
p->add_msg_if_player( _( "The geiger counter's scan LED turns on." ) );
it->convert( "geiger_on" ).active = true;
break;
default:
return 0;
}
return it->type->charges_to_use();
}
int iuse::teleport( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// That would be evil
return 0;
}
if( !it->ammo_sufficient() ) {
return 0;
}
p->moves -= to_turns<int>( 1_seconds );
g->teleport( p );
return it->type->charges_to_use();
}
int iuse::can_goo( player *p, item *it, bool, const tripoint & )
{
it->convert( "canister_empty" );
int tries = 0;
tripoint goop;
goop.z = p->posz();
do {
goop.x = p->posx() + rng( -2, 2 );
goop.y = p->posy() + rng( -2, 2 );
tries++;
} while( g->m.impassable( goop ) && tries < 10 );
if( tries == 10 ) {
return 0;
}
if( monster *const mon_ptr = g->critter_at<monster>( goop ) ) {
monster &critter = *mon_ptr;
if( g->u.sees( goop ) ) {
add_msg( _( "Black goo emerges from the canister and envelopes a %s!" ),
critter.name() );
}
critter.poly( mon_blob );
critter.set_speed_base( critter.get_speed_base() - rng( 5, 25 ) );
critter.set_hp( critter.get_speed() );
} else {
if( g->u.sees( goop ) ) {
add_msg( _( "Living black goo emerges from the canister!" ) );
}
if( monster *const goo = g->summon_mon( mon_blob, goop ) ) {
goo->friendly = -1;
}
}
tries = 0;
while( !one_in( 4 ) && tries < 10 ) {
tries = 0;
do {
goop.x = p->posx() + rng( -2, 2 );
goop.y = p->posy() + rng( -2, 2 );
tries++;
} while( g->m.impassable( goop ) &&
g->m.tr_at( goop ).is_null() && tries < 10 );
if( tries < 10 ) {
if( g->u.sees( goop ) ) {
add_msg( m_warning, _( "A nearby splatter of goo forms into a goo pit." ) );
}
g->m.trap_set( goop, tr_goo );
} else {
return 0;
}
}
return it->type->charges_to_use();
}
int iuse::throwable_extinguisher_act( player *, item *it, bool, const tripoint &pos )
{
if( pos.x == -999 || pos.y == -999 ) {
return 0;
}
if( g->m.get_field( pos, fd_fire ) != nullptr ) {
// Reduce the strength of fire (if any) in the target tile.
g->m.mod_field_intensity( pos, fd_fire, 0 - 1 );
// Slightly reduce the strength of fire around and in the target tile.
for( const tripoint &dest : g->m.points_in_radius( pos, 1 ) ) {
if( g->m.passable( dest ) && dest != pos ) {
g->m.mod_field_intensity( dest, fd_fire, 0 - rng( 0, 1 ) );
}
}
return 1;
}
it->active = false;
return 0;
}
int iuse::granade( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You pull the pin on the Granade." ) );
it->convert( "granade_act" );
it->charges = 5;
it->active = true;
return it->type->charges_to_use();
}
int iuse::granade_act( player *, item *it, bool t, const tripoint &pos )
{
if( pos.x == -999 || pos.y == -999 ) {
return 0;
}
if( t ) { // Simple timer effects
// Vol 0 = only heard if you hold it
sounds::sound( pos, 0, sounds::sound_t::speech, _( "Merged!" ), true, "speech", it->typeId() );
} else if( it->charges > 0 ) {
add_msg( m_info, _( "You've already pulled the %s's pin, try throwing it instead." ),
it->tname() );
return 0;
} else { // When that timer runs down...
int explosion_radius = 3;
int effect_roll = rng( 1, 5 );
auto buff_stat = [&]( int &current_stat, int modify_by ) {
auto modified_stat = current_stat + modify_by;
current_stat = std::max( current_stat, std::min( 15, modified_stat ) );
};
switch( effect_roll ) {
case 1:
sounds::sound( pos, 100, sounds::sound_t::speech, _( "BUGFIXES!" ), true, "speech", it->typeId() );
explosion_handler::draw_explosion( pos, explosion_radius, c_light_cyan );
for( const tripoint &dest : g->m.points_in_radius( pos, explosion_radius ) ) {
monster *const mon = g->critter_at<monster>( dest, true );
if( mon && ( mon->type->in_species( INSECT ) || mon->is_hallucination() ) ) {
mon->die_in_explosion( nullptr );
}
}
break;
case 2:
sounds::sound( pos, 100, sounds::sound_t::speech, _( "BUFFS!" ), true, "speech", it->typeId() );
explosion_handler::draw_explosion( pos, explosion_radius, c_green );
for( const tripoint &dest : g->m.points_in_radius( pos, explosion_radius ) ) {
if( monster *const mon_ptr = g->critter_at<monster>( dest ) ) {
monster &critter = *mon_ptr;
critter.set_speed_base(
critter.get_speed_base() * rng_float( 1.1, 2.0 ) );
critter.set_hp( critter.get_hp() * rng_float( 1.1, 2.0 ) );
} else if( npc *const person = g->critter_at<npc>( dest ) ) {
/** @EFFECT_STR_MAX increases possible granade str buff for NPCs */
buff_stat( person->str_max, rng( 0, person->str_max / 2 ) );
/** @EFFECT_DEX_MAX increases possible granade dex buff for NPCs */
buff_stat( person->dex_max, rng( 0, person->dex_max / 2 ) );
/** @EFFECT_INT_MAX increases possible granade int buff for NPCs */
buff_stat( person->int_max, rng( 0, person->int_max / 2 ) );
/** @EFFECT_PER_MAX increases possible granade per buff for NPCs */
buff_stat( person->per_max, rng( 0, person->per_max / 2 ) );
} else if( g->u.pos() == dest ) {
/** @EFFECT_STR_MAX increases possible granade str buff */
buff_stat( g->u.str_max, rng( 0, g->u.str_max / 2 ) );
/** @EFFECT_DEX_MAX increases possible granade dex buff */
buff_stat( g->u.dex_max, rng( 0, g->u.dex_max / 2 ) );
/** @EFFECT_INT_MAX increases possible granade int buff */
buff_stat( g->u.int_max, rng( 0, g->u.int_max / 2 ) );
/** @EFFECT_PER_MAX increases possible granade per buff */
buff_stat( g->u.per_max, rng( 0, g->u.per_max / 2 ) );
g->u.recalc_hp();
for( int part = 0; part < num_hp_parts; part++ ) {
g->u.hp_cur[part] *= 1 + rng( 0, 20 ) * .1;
if( g->u.hp_cur[part] > g->u.hp_max[part] ) {
g->u.hp_cur[part] = g->u.hp_max[part];
}
}
}
}
break;
case 3:
sounds::sound( pos, 100, sounds::sound_t::speech, _( "NERFS!" ), true, "speech", it->typeId() );
explosion_handler::draw_explosion( pos, explosion_radius, c_red );
for( const tripoint &dest : g->m.points_in_radius( pos, explosion_radius ) ) {
if( monster *const mon_ptr = g->critter_at<monster>( dest ) ) {
monster &critter = *mon_ptr;
critter.set_speed_base(
rng( 0, critter.get_speed_base() ) );
critter.set_hp( rng( 1, critter.get_hp() ) );
} else if( npc *const person = g->critter_at<npc>( dest ) ) {
/** @EFFECT_STR_MAX increases possible granade str debuff for NPCs (NEGATIVE) */
person->str_max -= rng( 0, person->str_max / 2 );
/** @EFFECT_DEX_MAX increases possible granade dex debuff for NPCs (NEGATIVE) */
person->dex_max -= rng( 0, person->dex_max / 2 );
/** @EFFECT_INT_MAX increases possible granade int debuff for NPCs (NEGATIVE) */
person->int_max -= rng( 0, person->int_max / 2 );
/** @EFFECT_PER_MAX increases possible granade per debuff for NPCs (NEGATIVE) */
person->per_max -= rng( 0, person->per_max / 2 );
} else if( g->u.pos() == dest ) {
/** @EFFECT_STR_MAX increases possible granade str debuff (NEGATIVE) */
g->u.str_max -= rng( 0, g->u.str_max / 2 );
/** @EFFECT_DEX_MAX increases possible granade dex debuff (NEGATIVE) */
g->u.dex_max -= rng( 0, g->u.dex_max / 2 );
/** @EFFECT_INT_MAX increases possible granade int debuff (NEGATIVE) */
g->u.int_max -= rng( 0, g->u.int_max / 2 );
/** @EFFECT_PER_MAX increases possible granade per debuff (NEGATIVE) */
g->u.per_max -= rng( 0, g->u.per_max / 2 );
g->u.recalc_hp();
for( int part = 0; part < num_hp_parts; part++ ) {
if( g->u.hp_cur[part] > 0 ) {
g->u.hp_cur[part] = rng( 1, g->u.hp_cur[part] );
}
}
}
}
break;
case 4:
sounds::sound( pos, 100, sounds::sound_t::speech, _( "REVERTS!" ), true, "speech", it->typeId() );
explosion_handler::draw_explosion( pos, explosion_radius, c_pink );
for( const tripoint &dest : g->m.points_in_radius( pos, explosion_radius ) ) {
if( monster *const mon_ptr = g->critter_at<monster>( dest ) ) {
monster &critter = *mon_ptr;
critter.set_speed_base( critter.type->speed );
critter.set_hp( critter.get_hp_max() );
critter.clear_effects();
} else if( npc *const person = g->critter_at<npc>( dest ) ) {
person->environmental_revert_effect();
} else if( g->u.pos() == dest ) {
g->u.environmental_revert_effect();
do_purify( g->u );
}
}
break;
case 5:
sounds::sound( pos, 100, sounds::sound_t::speech, _( "BEES!" ), true, "speech", it->typeId() );
explosion_handler::draw_explosion( pos, explosion_radius, c_yellow );
for( const tripoint &dest : g->m.points_in_radius( pos, explosion_radius ) ) {
if( one_in( 5 ) && !g->critter_at( dest ) ) {
g->m.add_field( dest, fd_bees, rng( 1, 3 ) );
}
}
break;
}
}
return it->type->charges_to_use();
}
int iuse::c4( player *p, item *it, bool, const tripoint & )
{
int time;
bool got_value = query_int( time, _( "Set the timer to (0 to cancel)?" ) );
if( !got_value || time <= 0 ) {
p->add_msg_if_player( _( "Never mind." ) );
return 0;
}
p->add_msg_if_player( _( "You set the timer to %d." ), time );
it->convert( "c4armed" );
it->charges = time;
it->active = true;
return it->type->charges_to_use();
}
int iuse::acidbomb_act( player *p, item *it, bool, const tripoint &pos )
{
if( !p->has_item( *it ) ) {
it->charges = -1;
for( const tripoint &tmp : g->m.points_in_radius( pos.x == -999 ? p->pos() : pos, 1 ) ) {
g->m.add_field( tmp, fd_acid, 3 );
}
return 1;
}
return 0;
}
int iuse::grenade_inc_act( player *p, item *it, bool t, const tripoint &pos )
{
if( pos.x == -999 || pos.y == -999 ) {
return 0;
}
if( t ) { // Simple timer effects
// Vol 0 = only heard if you hold it
sounds::sound( pos, 0, sounds::sound_t::alarm, _( "Tick!" ), true, "misc", "bomb_ticking" );
} else if( it->charges > 0 ) {
p->add_msg_if_player( m_info, _( "You've already released the handle, try throwing it instead." ) );
return 0;
} else { // blow up
int num_flames = rng( 3, 5 );
for( int current_flame = 0; current_flame < num_flames; current_flame++ ) {
tripoint dest( pos.x + rng( -5, 5 ), pos.y + rng( -5, 5 ), pos.z );
std::vector<tripoint> flames = line_to( pos, dest, 0, 0 );
for( auto &flame : flames ) {
g->m.add_field( flame, fd_fire, rng( 0, 2 ) );
}
}
explosion_handler::explosion( pos, 8, 0.8, true );
for( const tripoint &dest : g->m.points_in_radius( pos, 2 ) ) {
g->m.add_field( dest, fd_incendiary, 3 );
}
}
return 0;
}
int iuse::arrow_flammable( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( !p->use_charges_if_avail( "fire", 1 ) ) {
p->add_msg_if_player( m_info, _( "You need a source of fire!" ) );
return 0;
}
p->add_msg_if_player( _( "You light the arrow!" ) );
p->moves -= to_moves<int>( 1_seconds );
if( it->charges == 1 ) {
it->convert( "arrow_flamming" );
return 0;
}
item lit_arrow( *it );
lit_arrow.convert( "arrow_flamming" ).charges = 1;
p->i_add( lit_arrow );
return 1;
}
int iuse::molotov_lit( player *p, item *it, bool t, const tripoint &pos )
{
if( pos.x == -999 || pos.y == -999 ) {
return 0;
} else if( it->charges > 0 ) {
add_msg( m_info, _( "You've already lit the %s, try throwing it instead." ), it->tname() );
return 0;
} else if( p->has_item( *it ) && it->charges == 0 ) {
it->charges += 1;
if( one_in( 5 ) ) {
p->add_msg_if_player( _( "Your lit Molotov goes out." ) );
it->convert( "molotov" ).active = false;
}
} else {
if( !t ) {
for( auto &pt : g->m.points_in_radius( pos, 1, 0 ) ) {
const int intensity = 1 + one_in( 3 ) + one_in( 5 );
g->m.add_field( pt, fd_fire, intensity );
}
}
}
return 0;
}
int iuse::firecracker_pack( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( !p->has_charges( "fire", 1 ) ) {
p->add_msg_if_player( m_info, _( "You need a source of fire!" ) );
return 0;
}
p->add_msg_if_player( _( "You light the pack of firecrackers." ) );
it->convert( "firecracker_pack_act" );
it->charges = 26;
it->set_age( 0_turns );
it->active = true;
return 0; // don't use any charges at all. it has became a new item
}
int iuse::firecracker_pack_act( player *, item *it, bool, const tripoint &pos )
{
time_duration timer = it->age();
if( timer < 2_turns ) {
sounds::sound( pos, 0, sounds::sound_t::alarm, _( "ssss..." ), true, "misc", "lit_fuse" );
it->inc_damage();
} else if( it->charges > 0 ) {
int ex = rng( 4, 6 );
int i = 0;
if( ex > it->charges ) {
ex = it->charges;
}
for( i = 0; i < ex; i++ ) {
sounds::sound( pos, 20, sounds::sound_t::combat, _( "Bang!" ), false, "explosion", "small" );
}
it->charges -= ex;
}
if( it->charges == 0 ) {
it->charges = -1;
}
return 0;
}
int iuse::firecracker( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( !p->use_charges_if_avail( "fire", 1 ) ) {
p->add_msg_if_player( m_info, _( "You need a source of fire!" ) );
return 0;
}
p->add_msg_if_player( _( "You light the firecracker." ) );
it->convert( "firecracker_act" );
it->charges = 2;
it->active = true;
return it->type->charges_to_use();
}
int iuse::firecracker_act( player *, item *it, bool t, const tripoint &pos )
{
if( pos.x == -999 || pos.y == -999 ) {
return 0;
}
if( t ) { // Simple timer effects
sounds::sound( pos, 0, sounds::sound_t::alarm, _( "ssss..." ), true, "misc", "lit_fuse" );
} else if( it->charges > 0 ) {
add_msg( m_info, _( "You've already lit the %s, try throwing it instead." ), it->tname() );
return 0;
} else { // When that timer runs down...
sounds::sound( pos, 20, sounds::sound_t::combat, _( "Bang!" ), true, "explosion", "small" );
}
return 0;
}
int iuse::mininuke( player *p, item *it, bool, const tripoint & )
{
int time;
bool got_value = query_int( time, _( "Set the timer to ___ turns (0 to cancel)?" ) );
if( !got_value || time <= 0 ) {
p->add_msg_if_player( _( "Never mind." ) );
return 0;
}
p->add_msg_if_player( _( "You set the timer to %s." ),
to_string( time_duration::from_turns( time ) ) );
if( !p->is_npc() ) {
p->add_memorial_log( pgettext( "memorial_male", "Activated a mininuke." ),
pgettext( "memorial_female", "Activated a mininuke." ) );
}
it->convert( "mininuke_act" );
it->charges = time;
it->active = true;
return it->type->charges_to_use();
}
int iuse::pheromone( player *p, item *it, bool, const tripoint &pos )
{
if( !it->ammo_sufficient() ) {
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( pos.x == -999 || pos.y == -999 ) {
return 0;
}
p->add_msg_player_or_npc( _( "You squeeze the pheromone ball..." ),
_( "<npcname> squeezes the pheromone ball..." ) );
p->moves -= 15;
int converts = 0;
for( const tripoint &dest : g->m.points_in_radius( pos, 4 ) ) {
monster *const mon_ptr = g->critter_at<monster>( dest, true );
if( !mon_ptr ) {
continue;
}
monster &critter = *mon_ptr;
if( critter.type->in_species( ZOMBIE ) && critter.friendly == 0 &&
rng( 0, 500 ) > critter.get_hp() ) {
converts++;
critter.anger = 0;
}
}
if( g->u.sees( *p ) ) {
if( converts == 0 ) {
add_msg( _( "...but nothing happens." ) );
} else if( converts == 1 ) {
add_msg( m_good, _( "...and a nearby zombie becomes passive!" ) );
} else {
add_msg( m_good, _( "...and several nearby zombies become passive!" ) );
}
}
return it->type->charges_to_use();
}
int iuse::portal( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
return 0;
}
tripoint t( p->posx() + rng( -2, 2 ), p->posy() + rng( -2, 2 ), p->posz() );
g->m.trap_set( t, tr_portal );
return it->type->charges_to_use();
}
int iuse::tazer( player *p, item *it, bool, const tripoint &pos )
{
if( !it->units_sufficient( *p ) ) {
return 0;
}
tripoint pnt = pos;
if( pos == p->pos() ) {
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Shock where?" ) );
if( !pnt_ ) {
return 0;
}
pnt = *pnt_;
}
if( pnt == p->pos() ) {
p->add_msg_if_player( m_info, _( "Umm. No." ) );
return 0;
}
Creature *target = g->critter_at( pnt, true );
if( target == nullptr ) {
p->add_msg_if_player( _( "There's nothing to zap there!" ) );
return 0;
}
npc *foe = dynamic_cast<npc *>( target );
if( foe != nullptr &&
!foe->is_enemy() &&
!p->query_yn( _( "Really shock %s?" ), target->disp_name() ) ) {
return 0;
}
/** @EFFECT_DEX slightly increases chance of successfully using tazer */
/** @EFFECT_MELEE increases chance of successfully using a tazer */
int numdice = 3 + ( p->dex_cur / 2.5 ) + p->get_skill_level( skill_melee ) * 2;
p->moves -= to_turns<int>( 1_seconds );
/** @EFFECT_DODGE increases chance of dodging a tazer attack */
const bool tazer_was_dodged = dice( numdice, 10 ) < dice( target->get_dodge(), 10 );
if( tazer_was_dodged ) {
p->add_msg_player_or_npc( _( "You attempt to shock %s, but miss." ),
_( "<npcname> attempts to shock %s, but misses." ),
target->disp_name() );
} else {
// TODO: Maybe - Execute an attack and maybe zap something other than torso
// Maybe, because it's torso (heart) that fails when zapped with electricity
int dam = target->deal_damage( p, bp_torso, damage_instance( DT_ELECTRIC, rng( 5,
25 ) ) ).total_damage();
if( dam > 0 ) {
p->add_msg_player_or_npc( m_good,
_( "You shock %s!" ),
_( "<npcname> shocks %s!" ),
target->disp_name() );
} else {
p->add_msg_player_or_npc( m_warning,
_( "You unsuccessfully attempt to shock %s!" ),
_( "<npcname> unsuccessfully attempts to shock %s!" ),
target->disp_name() );
}
}
if( foe != nullptr ) {
foe->on_attacked( *p );
}
return it->type->charges_to_use();
}
int iuse::tazer2( player *p, item *it, bool b, const tripoint &pos )
{
if( it->ammo_remaining() >= 100 ) {
// Instead of having a ctrl+c+v of the function above, spawn a fake tazer and use it
// Ugly, but less so than copied blocks
item fake( "tazer", 0 );
fake.charges = 100;
return tazer( p, &fake, b, pos );
} else {
p->add_msg_if_player( m_info, _( "Insufficient power" ) );
}
return 0;
}
int iuse::shocktonfa_off( player *p, item *it, bool t, const tripoint &pos )
{
int choice = uilist( _( "tactical tonfa" ), {
_( "Zap something" ), _( "Turn on light" )
} );
switch( choice ) {
case 0: {
return iuse::tazer2( p, it, t, pos );
}
case 1: {
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_info, _( "The batteries are dead." ) );
return 0;
} else {
p->add_msg_if_player( _( "You turn the light on." ) );
it->convert( "shocktonfa_on" ).active = true;
return it->type->charges_to_use();
}
}
}
return 0;
}
int iuse::shocktonfa_on( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) { // Effects while simply on
} else {
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_info, _( "Your tactical tonfa is out of power." ) );
it->convert( "shocktonfa_off" ).active = false;
} else {
int choice = uilist( _( "tactical tonfa" ), {
_( "Zap something" ), _( "Turn off light" )
} );
switch( choice ) {
case 0: {
return iuse::tazer2( p, it, t, pos );
}
case 1: {
p->add_msg_if_player( _( "You turn off the light." ) );
it->convert( "shocktonfa_off" ).active = false;
}
}
}
}
return 0;
}
int iuse::mp3( player *p, item *it, bool, const tripoint & )
{
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_info, _( "The device's batteries are dead." ) );
} else if( p->has_active_item( "mp3_on" ) || p->has_active_item( "smartphone_music" ) ) {
p->add_msg_if_player( m_info, _( "You are already listening to music!" ) );
} else {
p->add_msg_if_player( m_info, _( "You put in the earbuds and start listening to music." ) );
if( it->typeId() == "mp3" ) {
it->convert( "mp3_on" ).active = true;
} else if( it->typeId() == "smart_phone" ) {
it->convert( "smartphone_music" ).active = true;
}
}
return it->type->charges_to_use();
}
static std::string get_music_description()
{
static const std::string no_description = _( "a sweet guitar solo!" );
static const std::string rare = _( "some bass-heavy post-glam speed polka." );
static const std::array<std::string, 5> descriptions = {{
_( "a sweet guitar solo!" ),
_( "a funky bassline." ),
_( "some amazing vocals." ),
_( "some pumping bass." ),
_( "dramatic classical music." )
}
};
if( one_in( 50 ) ) {
return rare;
}
size_t i = static_cast<size_t>( rng( 0, descriptions.size() * 2 ) );
if( i < descriptions.size() ) {
return descriptions[i];
}
// Not one of the hard-coded versions, let's apply a random string made up
// of snippets {a, b, c}, but only a 50% chance
// Actual chance = 24.5% of being selected
if( one_in( 2 ) ) {
const std::string &from_a = SNIPPET.random_from_category( "musicgenre_a" );
const std::string &from_b = SNIPPET.random_from_category( "musicgenre_b" );
const std::string &from_c = SNIPPET.random_from_category( "musicgenre_c" );
// Require all to be non-empty
if( !( from_a.empty() || from_b.empty() || from_c.empty() ) ) {
return from_a + from_b + from_c;
}
}
return no_description;
}
void iuse::play_music( player &p, const tripoint &source, const int volume, const int max_morale )
{
// TODO: what about other "player", e.g. when a NPC is listening or when the PC is listening,
// the other characters around should be able to profit as well.
const bool do_effects = p.can_hear( source, volume );
std::string sound = "music";
if( calendar::once_every( 5_minutes ) ) {
// Every 5 minutes, describe the music
const std::string music = get_music_description();
if( !music.empty() ) {
sound = music;
// descriptions aren't printed for sounds at our position
if( p.pos() == source && p.can_hear( source, volume ) ) {
p.add_msg_if_player( _( "You listen to %s" ), music );
}
}
}
// do not process mp3 player
if( volume != 0 ) {
sounds::ambient_sound( source, volume, sounds::sound_t::music, sound );
}
if( do_effects ) {
p.add_effect( effect_music, 1_turns );
p.add_morale( MORALE_MUSIC, 1, max_morale, 5_minutes, 2_minutes, true );
// mp3 player reduces hearing
if( volume == 0 ) {
p.add_effect( effect_earphones, 1_turns );
}
}
}
int iuse::mp3_on( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) { // Normal use
if( p->has_item( *it ) ) {
// mp3 player in inventory, we can listen
play_music( *p, pos, 0, 20 );
}
} else { // Turning it off
if( it->typeId() == "mp3_on" ) {
p->add_msg_if_player( _( "The mp3 player turns off." ) );
it->convert( "mp3" ).active = false;
} else if( it->typeId() == "smartphone_music" ) {
p->add_msg_if_player( _( "The phone turns off." ) );
it->convert( "smart_phone" ).active = false;
}
}
return it->type->charges_to_use();
}
int iuse::rpgdie( player *you, item *die, bool, const tripoint & )
{
int num_sides = die->get_var( "die_num_sides", 0 );
if( num_sides == 0 ) {
const std::vector<int> sides_options = { 4, 6, 8, 10, 12, 20, 50 };
const int sides = sides_options[ rng( 0, sides_options.size() - 1 ) ];
num_sides = sides;
die->set_var( "die_num_sides", sides );
}
const int roll = rng( 1, num_sides );
you->add_msg_if_player( _( "You roll a %d on your %d sided %s" ), roll, num_sides, die->tname() );
if( roll == num_sides ) {
add_msg( m_good, _( "Critical!" ) );
}
return roll;
}
int iuse::dive_tank( player *p, item *it, bool t, const tripoint & )
{
if( t ) { // Normal use
if( p->is_worn( *it ) ) {
if( p->is_underwater() && p->oxygen < 10 ) {
p->oxygen += 20;
}
if( one_in( 15 ) ) {
p->add_msg_if_player( m_bad, _( "You take a deep breath from your %s." ), it->tname() );
}
if( it->charges == 0 ) {
p->add_msg_if_player( m_bad, _( "Air in your %s runs out." ), it->tname() );
it->set_var( "overwrite_env_resist", 0 );
it->convert( it->typeId().substr( 0, it->typeId().size() - 3 ) ).active = false; // 3 = "_on"
}
} else { // not worn = off thanks to on-demand regulator
it->set_var( "overwrite_env_resist", 0 );
it->convert( it->typeId().substr( 0, it->typeId().size() - 3 ) ).active = false; // 3 = "_on"
}
} else { // Turning it on/off
if( it->charges == 0 ) {
p->add_msg_if_player( _( "Your %s is empty." ), it->tname() );
} else if( it->active ) { //off
p->add_msg_if_player( _( "You turn off the regulator and close the air valve." ) );
it->set_var( "overwrite_env_resist", 0 );
it->convert( it->typeId().substr( 0, it->typeId().size() - 3 ) ).active = false; // 3 = "_on"
} else { //on
if( !p->is_worn( *it ) ) {
p->add_msg_if_player( _( "You should wear it first." ) );
} else {
p->add_msg_if_player( _( "You turn on the regulator and open the air valve." ) );
it->set_var( "overwrite_env_resist", it->get_base_env_resist_w_filter() );
it->convert( it->typeId() + "_on" ).active = true;
}
}
}
if( it->charges == 0 ) {
it->set_var( "overwrite_env_resist", 0 );
it->convert( it->typeId().substr( 0, it->typeId().size() - 3 ) ).active = false; // 3 = "_on"
}
return it->type->charges_to_use();
}
int iuse::solarpack( player *p, item *it, bool, const tripoint & )
{
if( !p->has_bionic( bionic_id( "bio_cable" ) ) ) { // Cable CBM required
p->add_msg_if_player(
_( "You have no cable charging system to plug it in, so you leave it alone." ) );
return 0;
} else if( !p->has_active_bionic( bionic_id( "bio_cable" ) ) ) { // when OFF it takes no effect
p->add_msg_if_player( _( "Activate your cable charging system to take advantage of it." ) );
}
if( it->is_armor() && !( p->is_worn( *it ) ) ) {
p->add_msg_if_player( m_neutral, _( "You need to wear the %1$s before you can unfold it." ),
it->tname() );
return 0;
}
// no doubled sources of power
if( p->is_wearing( "solarpack_on" ) || p->is_wearing( "q_solarpack_on" ) ) {
p->add_msg_if_player( m_neutral, _( "You cannot use the %1$s with another of it's kind." ),
it->tname() );
return 0;
}
p->add_msg_if_player(
_( "You unfold solar array from the pack. You still need to connect it with a cable." ) );
if( it->typeId() == "solarpack" ) {
it->convert( "solarpack_on" );
} else {
it->convert( "q_solarpack_on" );
}
return 0;
}
int iuse::solarpack_off( player *p, item *it, bool, const tripoint & )
{
if( !p->is_worn( *it ) ) { // folding when not worn
p->add_msg_if_player( _( "You fold your portable solar array into the pack." ) );
} else {
p->add_msg_if_player( _( "You unplug and fold your portable solar array into the pack." ) );
}
if( it->typeId() == "solarpack_on" ) {
it->convert( "solarpack" );
} else {
it->convert( "q_solarpack" );
}
return 0;
}
int iuse::gasmask( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) { // Normal use
if( p->is_worn( *it ) ) {
// calculate amount of absorbed gas per filter charge
const field &gasfield = g->m.field_at( pos );
for( auto &dfield : gasfield ) {
const field_entry &entry = dfield.second;
const field_type_id fid = entry.get_field_type();
if( fid == fd_smoke ) {
it->set_var( "gas_absorbed", it->get_var( "gas_absorbed", 0 ) + 12 );
}
if( fid == fd_tear_gas || fid == fd_toxic_gas || fid == fd_gas_vent ||
fid == fd_smoke_vent || fid == fd_relax_gas || fid == fd_fungal_haze ) {
it->set_var( "gas_absorbed", it->get_var( "gas_absorbed", 0 ) + 15 );
}
}
if( it->get_var( "gas_absorbed", 0 ) >= 100 ) {
it->ammo_consume( 1, p->pos() );
it->set_var( "gas_absorbed", 0 );
}
if( it->charges == 0 ) {
p->add_msg_player_or_npc(
m_bad,
_( "Your %s requires new filter!" ),
_( "<npcname> needs new gas mask filter!" )
, it->tname() );
}
}
} else { // activate
if( it->charges == 0 ) {
p->add_msg_if_player( _( "Your %s don't have a filter." ), it->tname() );
} else {
p->add_msg_if_player( _( "You prepared your %s." ), it->tname() );
it->active = true;
it->set_var( "overwrite_env_resist", it->get_base_env_resist_w_filter() );
}
}
if( it->charges == 0 ) {
it->set_var( "overwrite_env_resist", 0 );
it->active = false;
}
return it->type->charges_to_use();
}
int iuse::portable_game( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// Long action
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( p->has_trait( trait_ILLITERATE ) ) {
add_msg( _( "You're illiterate!" ) );
return 0;
} else if( it->ammo_remaining() < 15 ) {
p->add_msg_if_player( m_info, _( "The %s's batteries are dead." ), it->tname() );
return 0;
} else {
std::string loaded_software = "robot_finds_kitten";
uilist as_m;
as_m.text = _( "What do you want to play?" );
as_m.entries.emplace_back( 1, true, '1', _( "Robot finds Kitten" ) );
as_m.entries.emplace_back( 2, true, '2', _( "S N A K E" ) );
as_m.entries.emplace_back( 3, true, '3', _( "Sokoban" ) );
as_m.entries.emplace_back( 4, true, '4', _( "Minesweeper" ) );
as_m.entries.emplace_back( 5, true, '5', _( "Lights on!" ) );
as_m.query();
switch( as_m.ret ) {
case 1:
loaded_software = "robot_finds_kitten";
break;
case 2:
loaded_software = "snake_game";
break;
case 3:
loaded_software = "sokoban_game";
break;
case 4:
loaded_software = "minesweeper_game";
break;
case 5:
loaded_software = "lightson_game";
break;
default: //Cancel
return 0;
}
//Play in 15-minute chunks
int time = to_turns<int>( 15_minutes );
p->add_msg_if_player( _( "You play on your %s for a while." ), it->tname() );
p->assign_activity( activity_id( "ACT_GAME" ), time, -1, p->get_item_position( it ), "gaming" );
std::map<std::string, std::string> game_data;
game_data.clear();
int game_score = 0;
play_videogame( loaded_software, game_data, game_score );
if( game_data.find( "end_message" ) != game_data.end() ) {
p->add_msg_if_player( game_data["end_message"] );
}
if( game_score != 0 ) {
if( game_data.find( "moraletype" ) != game_data.end() ) {
std::string moraletype = game_data.find( "moraletype" )->second;
if( moraletype == "MORALE_GAME_FOUND_KITTEN" ) {
p->add_morale( MORALE_GAME_FOUND_KITTEN, game_score, 110 );
} /*else if ( ...*/
} else {
p->add_morale( MORALE_GAME, game_score, 110 );
}
}
}
return it->type->charges_to_use();
}
int iuse::hand_crank( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// Long action
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "It's not waterproof enough to work underwater." ) );
return 0;
}
if( p->get_fatigue() >= DEAD_TIRED ) {
p->add_msg_if_player( m_info, _( "You're too exhausted to keep cranking." ) );
return 0;
}
item *magazine = it->magazine_current();
if( magazine && magazine->has_flag( "RECHARGE" ) ) {
// 1600 minutes. It shouldn't ever run this long, but it's an upper bound.
// expectation is it runs until the player is too tired.
int time = to_moves<int>( 1600_minutes );
if( it->ammo_capacity() > it->ammo_remaining() ) {
p->add_msg_if_player( string_format( _( "You start cranking the %s to charge its %s." ),
it->tname(), it->magazine_current()->tname() ) ) ;
p->assign_activity( activity_id( "ACT_HAND_CRANK" ), time, -1, p->get_item_position( it ),
"hand-cranking" );
} else {
p->add_msg_if_player( string_format(
_( "You could use the %s to charge its %s, but it's already charged." ), it->tname(),
magazine->tname() ) ) ;
}
} else {
p->add_msg_if_player( m_info, _( "You need a rechargeable battery cell to charge." ) );
}
return 0;
}
int iuse::vibe( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// Long action
// Also, that would be creepy as fuck, seriously
return 0;
}
if( ( p->is_underwater() ) && ( !( ( p->has_trait( trait_GILLS ) ) ||
( p->is_wearing( "rebreather_on" ) ) ||
( p->is_wearing( "rebreather_xl_on" ) ) || ( p->is_wearing( "mask_h20survivor_on" ) ) ) ) ) {
p->add_msg_if_player( m_info, _( "It's waterproof, but oxygen maybe?" ) );
return 0;
}
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_info, _( "The %s's batteries are dead." ), it->tname() );
return 0;
}
if( p->get_fatigue() >= DEAD_TIRED ) {
p->add_msg_if_player( m_info, _( "*Your* batteries are dead." ) );
return 0;
} else {
int time = to_turns<int>( 20_minutes );
if( it->ammo_remaining() > 0 ) {
p->add_msg_if_player( _( "You fire up your %s and start getting the tension out." ),
it->tname() );
} else {
p->add_msg_if_player( _( "You whip out your %s and start getting the tension out." ),
it->tname() );
}
p->assign_activity( activity_id( "ACT_VIBE" ), time, -1, p->get_item_position( it ),
"de-stressing" );
}
return it->type->charges_to_use();
}
int iuse::vortex( player *p, item *it, bool, const tripoint & )
{
std::vector<tripoint> spawn;
auto empty_add = [&]( int x, int y ) {
tripoint pt( x, y, p->posz() );
if( g->is_empty( pt ) ) {
spawn.push_back( pt );
}
};
for( int i = -3; i <= 3; i++ ) {
empty_add( p->posx() - 3, p->posy() + i );
empty_add( p->posx() + 3, p->posy() + i );
empty_add( p->posx() + i, p->posy() - 3 );
empty_add( p->posx() + i, p->posy() + 3 );
}
if( spawn.empty() ) {
p->add_msg_if_player( m_warning, _( "Air swirls around you for a moment." ) );
return it->convert( "spiral_stone" ).type->charges_to_use();
}
p->add_msg_if_player( m_warning, _( "Air swirls all over..." ) );
p->moves -= to_turns<int>( 1_seconds );
it->convert( "spiral_stone" );
monster mvortex( mon_vortex, random_entry( spawn ) );
mvortex.friendly = -1;
g->add_zombie( mvortex );
return it->type->charges_to_use();
}
int iuse::dog_whistle( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
p->add_msg_if_player( _( "You blow your dog whistle." ) );
for( monster &critter : g->all_monsters() ) {
if( critter.friendly != 0 && critter.has_flag( MF_DOGFOOD ) ) {
bool u_see = g->u.sees( critter );
if( critter.has_effect( effect_docile ) ) {
if( u_see ) {
p->add_msg_if_player( _( "Your %s looks ready to attack." ), critter.name() );
}
critter.remove_effect( effect_docile );
} else {
if( u_see ) {
p->add_msg_if_player( _( "Your %s goes docile." ), critter.name() );
}
critter.add_effect( effect_docile, 1_turns, num_bp, true );
}
}
}
return it->type->charges_to_use();
}
int iuse::blood_draw( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
return 0; // No NPCs for now!
}
if( !it->contents.empty() ) {
p->add_msg_if_player( m_info, _( "That %s is full!" ), it->tname() );
return 0;
}
item blood( "blood", calendar::turn );
bool drew_blood = false;
bool acid_blood = false;
for( auto &map_it : g->m.i_at( p->posx(), p->posy() ) ) {
if( map_it.is_corpse() &&
query_yn( _( "Draw blood from %s?" ),
colorize( map_it.tname(), map_it.color_in_inventory() ) ) ) {
p->add_msg_if_player( m_info, _( "You drew blood from the %s..." ), map_it.tname() );
drew_blood = true;
auto bloodtype( map_it.get_mtype()->bloodType() );
if( bloodtype == fd_acid ) {
acid_blood = true;
} else {
blood.set_mtype( map_it.get_mtype() );
}
}
}
if( !drew_blood && query_yn( _( "Draw your own blood?" ) ) ) {
p->add_msg_if_player( m_info, _( "You drew your own blood..." ) );
drew_blood = true;
if( p->has_trait( trait_ACIDBLOOD ) ) {
acid_blood = true;
}
p->mod_stored_nutr( 10 );
p->mod_thirst( 10 );
p->mod_pain( 3 );
}
if( acid_blood ) {
item acid( "acid", calendar::turn );
it->put_in( acid );
if( one_in( 3 ) ) {
if( it->inc_damage( DT_ACID ) ) {
p->add_msg_if_player( m_info, _( "...but acidic blood melts the %s, destroying it!" ),
it->tname() );
p->i_rem( it );
return 0;
}
p->add_msg_if_player( m_info, _( "...but acidic blood damages the %s!" ), it->tname() );
}
return it->type->charges_to_use();
}
if( !drew_blood ) {
return it->type->charges_to_use();
}
it->put_in( blood );
return it->type->charges_to_use();
}
//This is just used for robofac_intercom_mission_2
int iuse::mind_splicer( player *p, item *it, bool, const tripoint & )
{
for( auto &map_it : g->m.i_at( p->posx(), p->posy() ) ) {
if( map_it.typeId() == "rmi2_corpse" &&
query_yn( _( "Use the mind splicer kit on the %s?" ), colorize( map_it.tname(),
map_it.color_in_inventory() ) ) ) {
int pos = g->inv_for_id( itype_id( "data_card" ), _( "Select storage media" ) );
item &data_card = p->i_at( pos );
if( data_card.is_null() ) {
add_msg( m_info, _( "Nevermind." ) );
return 0;
}
///\EFFECT_DEX makes using the mind splicer faster
///\EFFECT_FIRSTAID makes using the mind splicer faster
const time_duration time = std::max( 150_minutes - 20_minutes * ( p->get_skill_level(
skill_firstaid ) - 1 ) - 10_minutes * ( p->get_dex() - 8 ), 30_minutes );
player_activity act( activity_id( "ACT_MIND_SPLICER" ), to_moves<int>( time ) );
act.targets.push_back( item_location( *p, &data_card ) );
p->assign_activity( act );
return it->type->charges_to_use();
}
}
add_msg( m_info, _( "There's nothing to use the %s on here." ), it->tname() );
return 0;
}
void iuse::cut_log_into_planks( player &p )
{
p.moves -= to_turns<int>( 3_seconds );
p.add_msg_if_player( _( "You cut the log into planks." ) );
const int max_planks = 10;
/** @EFFECT_FABRICATION increases number of planks cut from a log */
int planks = normal_roll( 2 + p.get_skill_level( skill_fabrication ), 1 );
int wasted_planks = max_planks - planks;
int scraps = rng( wasted_planks, wasted_planks * 3 );
planks = std::min( planks, max_planks );
if( planks > 0 ) {
item plank( "2x4", calendar::turn );
p.i_add_or_drop( plank, planks );
p.add_msg_if_player( m_good, _( "You produce %d planks." ), planks );
}
if( scraps > 0 ) {
item scrap( "splinter", calendar::turn );
p.i_add_or_drop( scrap, scraps );
p.add_msg_if_player( m_good, _( "You produce %d splinters." ), scraps );
}
if( planks < max_planks / 2 ) {
add_msg( m_bad, _( "You waste a lot of the wood." ) );
}
}
int iuse::lumber( player *p, item *it, bool t, const tripoint & )
{
if( t ) {
return 0;
}
// Check if player is standing on any lumber
for( auto &i : g->m.i_at( p->pos() ) ) {
if( i.typeId() == "log" ) {
g->m.i_rem( p->pos(), &i );
cut_log_into_planks( *p );
return it->type->charges_to_use();
}
}
// If the player is not standing on a log, check inventory
int pos = g->inv_for_id( itype_id( "log" ), _( "Cut up what?" ) );
item &cut = p->i_at( pos );
if( cut.is_null() ) {
add_msg( m_info, _( "You do not have that item!" ) );
return 0;
}
p->i_rem( &cut );
cut_log_into_planks( *p );
return it->type->charges_to_use();
}
static int chop_moves( player *p, item *it )
{
// quality of tool
const int quality = it->get_quality( AXE );
// attribute; regular tools - based on STR, powered tools - based on DEX
const int attr = it->has_flag( "POWERED" ) ? p->dex_cur : p->str_cur;
int moves = to_moves<int>( time_duration::from_minutes( 60 - attr ) / std::pow( 2, quality - 1 ) );
const int helpersize = g->u.get_num_crafting_helpers( 3 );
moves = moves * ( 1 - ( helpersize / 10 ) );
return moves;
}
int iuse::chop_tree( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Chop down which tree?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
add_msg( m_info, _( "You're not stern enough to shave yourself with THIS." ) );
return 0;
}
int moves;
if( g->m.has_flag( "TREE", pnt ) ) {
moves = chop_moves( p, it );
} else {
add_msg( m_info, _( "You can't chop down that." ) );
return 0;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
p->assign_activity( activity_id( "ACT_CHOP_TREE" ), moves, -1, p->get_item_position( it ) );
p->activity.placement = pnt;
return it->type->charges_to_use();
}
int iuse::chop_logs( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Chop which tree trunk?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
int moves;
const ter_id ter = g->m.ter( pnt );
if( ter == t_trunk || ter == t_stump ) {
moves = chop_moves( p, it );
} else {
add_msg( m_info, _( "You can't chop that." ) );
return 0;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
p->assign_activity( activity_id( "ACT_CHOP_LOGS" ), moves, -1, p->get_item_position( it ) );
p->activity.placement = pnt;
return it->type->charges_to_use();
}
int iuse::oxytorch( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// Long action
return 0;
}
static const quality_id GLARE( "GLARE" );
if( !p->has_quality( GLARE, 2 ) ) {
add_msg( m_info, _( "You need welding goggles to do that." ) );
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Cut up metal where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
add_msg( m_info, _( "Yuck. Acetylene gas smells weird." ) );
return 0;
}
const ter_id ter = g->m.ter( pnt );
const auto furn = g->m.furn( pnt );
int turns = 0;
if( furn == f_rack || ter == t_chainfence_posts ) {
turns = to_turns<int>( 2_seconds );
} else if( ter == t_window_enhanced || ter == t_window_enhanced_noglass ) {
turns = to_turns<int>( 5_seconds );
} else if( ter == t_chainfence || ter == t_chaingate_c ||
ter == t_chaingate_l || ter == t_bars || ter == t_window_bars_alarm ||
ter == t_window_bars || ter == t_reb_cage ) {
turns = to_turns<int>( 10_seconds );
} else if( ter == t_door_metal_locked || ter == t_door_metal_c || ter == t_door_bar_c ||
ter == t_door_bar_locked || ter == t_door_metal_pickable ) {
turns = to_turns<int>( 15_seconds );
} else {
add_msg( m_info, _( "You can't cut that." ) );
return 0;
}
const int charges = turns * it->ammo_required();
if( charges > it->ammo_remaining() ) {
add_msg( m_info, _( "Your torch doesn't have enough acetylene to cut that." ) );
return 0;
}
// placing ter here makes resuming tasks work better
p->assign_activity( activity_id( "ACT_OXYTORCH" ), turns, static_cast<int>( ter ),
p->get_item_position( it ) );
p->activity.placement = pnt;
p->activity.values.push_back( charges );
// charges will be consumed in oxytorch_do_turn, not here
return 0;
}
int iuse::hacksaw( player *p, item *it, bool t, const tripoint & )
{
if( !p || t ) {
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Cut up metal where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
add_msg( m_info, _( "Why would you do that?" ) );
add_msg( m_info, _( "You're not even chained to a boiler." ) );
return 0;
}
const ter_id ter = g->m.ter( pnt );
int moves;
if( ter == t_chainfence_posts || g->m.furn( pnt ) == f_rack ) {
moves = to_moves<int>( 2_minutes );
} else if( ter == t_window_enhanced || ter == t_window_enhanced_noglass ) {
moves = to_moves<int>( 5_minutes );
} else if( ter == t_chainfence || ter == t_chaingate_c ||
ter == t_chaingate_l || ter == t_window_bars_alarm || ter == t_window_bars || ter == t_reb_cage ) {
moves = to_moves<int>( 10_minutes );
} else if( ter == t_door_bar_c || ter == t_door_bar_locked || ter == t_bars ) {
moves = to_moves<int>( 15_minutes );
} else {
add_msg( m_info, _( "You can't cut that." ) );
return 0;
}
p->assign_activity( activity_id( "ACT_HACKSAW" ), moves, static_cast<int>( ter ),
p->get_item_position( it ) );
p->activity.placement = pnt;
return it->type->charges_to_use();
}
int iuse::boltcutters( player *p, item *it, bool, const tripoint & )
{
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Cut up metal where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
p->add_msg_if_player(
_( "You neatly sever all of the veins and arteries in your body. Oh wait, Never mind." ) );
return 0;
}
if( g->m.ter( pnt ) == t_chaingate_l ) {
p->moves -= to_turns<int>( 1_seconds );
g->m.ter_set( pnt, t_chaingate_c );
sounds::sound( pnt, 5, sounds::sound_t::combat, _( "Gachunk!" ), true, "tool", "boltcutters" );
g->m.spawn_item( p->posx(), p->posy(), "scrap", 3 );
} else if( g->m.ter( pnt ) == t_chainfence ) {
p->moves -= to_turns<int>( 5_seconds );
g->m.ter_set( pnt, t_chainfence_posts );
sounds::sound( pnt, 5, sounds::sound_t::combat, _( "Snick, snick, gachunk!" ), true, "tool",
"boltcutters" );
g->m.spawn_item( pnt, "wire", 20 );
} else {
add_msg( m_info, _( "You can't cut that." ) );
return 0;
}
return it->type->charges_to_use();
}
int iuse::mop( player *p, item *it, bool, const tripoint & )
{
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Mop where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( pnt == p->pos() ) {
p->add_msg_if_player( _( "You mop yourself up." ) );
p->add_msg_if_player( _( "The universe implodes and reforms around you." ) );
return 0;
}
if( p->is_blind() ) {
add_msg( _( "You move the mop around, unsure whether it's doing any good." ) );
p->moves -= 15;
if( one_in( 3 ) ) {
g->m.mop_spills( pnt );
}
} else if( g->m.mop_spills( pnt ) ) {
add_msg( _( "You mop up the spill." ) );
p->moves -= 15;
} else {
p->add_msg_if_player( m_info, _( "There's nothing to mop there." ) );
return 0;
}
return it->type->charges_to_use();
}
int iuse::artifact( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ) {
// TODO: Allow this for trusting NPCs
return 0;
}
if( !it->is_artifact() ) {
debugmsg( "iuse::artifact called on a non-artifact item! %s",
it->tname() );
return 0;
} else if( !it->is_tool() ) {
debugmsg( "iuse::artifact called on a non-tool artifact! %s",
it->tname() );
return 0;
}
if( !p->is_npc() ) {
//~ %s is artifact name
p->add_memorial_log( pgettext( "memorial_male", "Activated the %s." ),
pgettext( "memorial_female", "Activated the %s." ),
it->tname( 1, false ) );
}
const auto &art = it->type->artifact;
size_t num_used = rng( 1, art->effects_activated.size() );
if( num_used < art->effects_activated.size() ) {
num_used += rng( 1, art->effects_activated.size() - num_used );
}
std::vector<art_effect_active> effects = art->effects_activated;
for( size_t i = 0; i < num_used && !effects.empty(); i++ ) {
const art_effect_active used = random_entry_removed( effects );
switch( used ) {
case AEA_STORM: {
sounds::sound( p->pos(), 10, sounds::sound_t::combat, _( "Ka-BOOM!" ), true, "environment",
"thunder_near" );
int num_bolts = rng( 2, 4 );
for( int j = 0; j < num_bolts; j++ ) {
int xdir = 0;
int ydir = 0;
while( xdir == 0 && ydir == 0 ) {
xdir = rng( -1, 1 );
ydir = rng( -1, 1 );
}
int dist = rng( 4, 12 );
int boltx = p->posx(), bolty = p->posy();
for( int n = 0; n < dist; n++ ) {
boltx += xdir;
bolty += ydir;
g->m.add_field( {boltx, bolty, p->posz()}, fd_electricity, rng( 2, 3 ) );
if( one_in( 4 ) ) {
if( xdir == 0 ) {
xdir = rng( 0, 1 ) * 2 - 1;
} else {
xdir = 0;
}
}
if( one_in( 4 ) ) {
if( ydir == 0 ) {
ydir = rng( 0, 1 ) * 2 - 1;
} else {
ydir = 0;
}
}
}
}
}
break;
case AEA_FIREBALL: {
if( const cata::optional<tripoint> fireball = g->look_around() ) {
explosion_handler::explosion( *fireball, 180, 0.5, true );
}
}
break;
case AEA_ADRENALINE:
p->add_msg_if_player( m_good, _( "You're filled with a roaring energy!" ) );
p->add_effect( effect_adrenaline, rng( 20_minutes, 25_minutes ) );
break;
case AEA_MAP: {
const tripoint center = p->global_omt_location();
const bool new_map = overmap_buffer.reveal(
point( center.x, center.y ), 20, center.z );
if( new_map ) {
p->add_msg_if_player( m_warning, _( "You have a vision of the surrounding area..." ) );
p->moves -= to_turns<int>( 1_seconds );
}
}
break;
case AEA_BLOOD: {
bool blood = false;
for( const tripoint &tmp : g->m.points_in_radius( p->pos(), 4 ) ) {
if( !one_in( 4 ) && g->m.add_field( tmp, fd_blood, 3 ) &&
( blood || g->u.sees( tmp ) ) ) {
blood = true;
}
}
if( blood ) {
p->add_msg_if_player( m_warning, _( "Blood soaks out of the ground and walls." ) );
}
}
break;
case AEA_FATIGUE: {
p->add_msg_if_player( m_warning, _( "The fabric of space seems to decay." ) );
int x = rng( p->posx() - 3, p->posx() + 3 ), y = rng( p->posy() - 3, p->posy() + 3 );
g->m.add_field( {x, y, p->posz()}, fd_fatigue, rng( 1, 2 ) );
}
break;
case AEA_ACIDBALL: {
if( const cata::optional<tripoint> acidball = g->look_around() ) {
for( const tripoint &tmp : g->m.points_in_radius( *acidball, 1 ) ) {
g->m.add_field( tmp, fd_acid, rng( 2, 3 ) );
}
}
}
break;
case AEA_PULSE:
sounds::sound( p->pos(), 30, sounds::sound_t::combat, _( "The earth shakes!" ), true, "misc",
"earthquake" );
for( const tripoint &pt : g->m.points_in_radius( p->pos(), 2 ) ) {
g->m.bash( pt, 40 );
g->m.bash( pt, 40 ); // Multibash effect, so that doors &c will fall
g->m.bash( pt, 40 );
if( g->m.is_bashable( pt ) && rng( 1, 10 ) >= 3 ) {
g->m.bash( pt, 999, false, true );
}
}
break;
case AEA_HEAL:
p->add_msg_if_player( m_good, _( "You feel healed." ) );
p->healall( 2 );
break;
case AEA_CONFUSED:
for( const tripoint &dest : g->m.points_in_radius( p->pos(), 8 ) ) {
if( monster *const mon = g->critter_at<monster>( dest, true ) ) {
mon->add_effect( effect_stunned, rng( 5_turns, 15_turns ) );
}
}
break;
case AEA_ENTRANCE:
for( const tripoint &dest : g->m.points_in_radius( p->pos(), 8 ) ) {
monster *const mon = g->critter_at<monster>( dest, true );
if( mon && mon->friendly == 0 && rng( 0, 600 ) > mon->get_hp() ) {
mon->make_friendly();
}
}
break;
case AEA_BUGS: {
int roll = rng( 1, 10 );
mtype_id bug = mtype_id::NULL_ID();
int num = 0;
std::vector<tripoint> empty;
for( const tripoint &dest : g->m.points_in_radius( p->pos(), 1 ) ) {
if( g->is_empty( dest ) ) {
empty.push_back( dest );
}
}
if( empty.empty() || roll <= 4 ) {
p->add_msg_if_player( m_warning, _( "Flies buzz around you." ) );
} else if( roll <= 7 ) {
p->add_msg_if_player( m_warning, _( "Giant flies appear!" ) );
bug = mon_fly;
num = rng( 2, 4 );
} else if( roll <= 9 ) {
p->add_msg_if_player( m_warning, _( "Giant bees appear!" ) );
bug = mon_bee;
num = rng( 1, 3 );
} else {
p->add_msg_if_player( m_warning, _( "Giant wasps appear!" ) );
bug = mon_wasp;
num = rng( 1, 2 );
}
if( bug ) {
for( int j = 0; j < num && !empty.empty(); j++ ) {
const tripoint spawnp = random_entry_removed( empty );
if( monster *const b = g->summon_mon( bug, spawnp ) ) {
b->friendly = -1;
b->add_effect( effect_pet, 1_turns, num_bp, true );
}
}
}
}
break;
case AEA_TELEPORT:
g->teleport( p );
break;
case AEA_LIGHT:
p->add_msg_if_player( _( "The %s glows brightly!" ), it->tname() );
g->events.add( EVENT_ARTIFACT_LIGHT, calendar::turn + 3_minutes );
break;
case AEA_GROWTH: {
monster tmptriffid( mtype_id::NULL_ID(), p->pos() );
mattack::growplants( &tmptriffid );
}
break;
case AEA_HURTALL:
for( monster &critter : g->all_monsters() ) {
critter.apply_damage( nullptr, bp_torso, rng( 0, 5 ) );
}
break;
case AEA_RADIATION:
add_msg( m_warning, _( "Horrible gases are emitted!" ) );
for( const tripoint &dest : g->m.points_in_radius( p->pos(), 1 ) ) {
g->m.add_field( dest, fd_nuke_gas, rng( 2, 3 ) );
}
break;
case AEA_PAIN:
p->add_msg_if_player( m_bad, _( "You're wracked with pain!" ) );
// OK, the Lovecraftian thingamajig can bring Deadened
// masochists & Cenobites the stimulation they've been
// craving ;)
p->mod_pain_noresist( rng( 5, 15 ) );
break;
case AEA_MUTATE:
if( !one_in( 3 ) ) {
p->mutate();
}
break;
case AEA_PARALYZE:
p->add_msg_if_player( m_bad, _( "You're paralyzed!" ) );
p->moves -= rng( 50, 200 );
break;
case AEA_FIRESTORM: {
p->add_msg_if_player( m_bad, _( "Fire rains down around you!" ) );
std::vector<tripoint> ps = closest_tripoints_first( 3, p->pos() );
for( auto p_it : ps ) {
if( !one_in( 3 ) ) {
g->m.add_field( p_it, fd_fire, 1 + rng( 0, 1 ) * rng( 0, 1 ), 3_minutes );
}
}
break;
}
case AEA_ATTENTION:
p->add_msg_if_player( m_warning, _( "You feel like your action has attracted attention." ) );
p->add_effect( effect_attention, rng( 1_hours, 3_hours ) );
break;
case AEA_TELEGLOW:
p->add_msg_if_player( m_warning, _( "You feel unhinged." ) );
p->add_effect( effect_teleglow, rng( 30_minutes, 120_minutes ) );
break;
case AEA_NOISE:
sounds::sound( p->pos(), 100, sounds::sound_t::combat,
string_format( _( "a deafening boom from %s %s" ),
p->disp_name( true ), it->tname() ), true, "misc", "shockwave" );
break;
case AEA_SCREAM:
sounds::sound( p->pos(), 40, sounds::sound_t::alert,
string_format( _( "a disturbing scream from %s %s" ),
p->disp_name( true ), it->tname() ), true, "shout", "scream" );
if( !p->is_deaf() ) {
p->add_morale( MORALE_SCREAM, -10, 0, 30_minutes, 1_minutes );
}
break;
case AEA_DIM:
p->add_msg_if_player( _( "The sky starts to dim." ) );
g->events.add( EVENT_DIM, calendar::turn + 5_minutes );
break;
case AEA_FLASH:
p->add_msg_if_player( _( "The %s flashes brightly!" ), it->tname() );
explosion_handler::flashbang( p->pos() );
break;
case AEA_VOMIT:
p->add_msg_if_player( m_bad, _( "A wave of nausea passes through you!" ) );
p->vomit();
break;
case AEA_SHADOWS: {
int num_shadows = rng( 4, 8 );
int num_spawned = 0;
for( int j = 0; j < num_shadows; j++ ) {
int tries = 0;
tripoint monp = p->pos();
do {
if( one_in( 2 ) ) {
monp.x = rng( p->posx() - 5, p->posx() + 5 );
monp.y = ( one_in( 2 ) ? p->posy() - 5 : p->posy() + 5 );
} else {
monp.x = ( one_in( 2 ) ? p->posx() - 5 : p->posx() + 5 );
monp.y = rng( p->posy() - 5, p->posy() + 5 );
}
} while( tries < 5 && !g->is_empty( monp ) &&
!g->m.sees( monp, p->pos(), 10 ) );
if( tries < 5 ) { // TODO: tries increment is missing, so this expression is always true
if( monster *const spawned = g->summon_mon( mon_shadow, monp ) ) {
num_spawned++;
spawned->reset_special_rng( "DISAPPEAR" );
}
}
}
if( num_spawned > 1 ) {
p->add_msg_if_player( m_warning, _( "Shadows form around you." ) );
} else if( num_spawned == 1 ) {
p->add_msg_if_player( m_warning, _( "A shadow forms nearby." ) );
}
}
break;
case AEA_STAMINA_EMPTY:
p->add_msg_if_player( m_bad, _( "Your body feels like jelly." ) );
p->stamina = p->stamina * 1 / ( rng( 3, 8 ) );
break;
case AEA_FUN:
p->add_msg_if_player( m_good, _( "You're filled with euphoria!" ) );
p->add_morale( MORALE_FEELING_GOOD, rng( 20, 50 ), 0, 5_minutes, 5_turns, false );
break;
case AEA_SPLIT: // TODO: Add something
break;
case AEA_NULL: // BUG
case NUM_AEAS:
default:
debugmsg( "iuse::artifact(): wrong artifact type (%d)", used );
break;
}
}
return it->type->charges_to_use();
}
int iuse::spray_can( player *p, item *it, bool, const tripoint & )
{
return handle_ground_graffiti( *p, it, _( "Spray what?" ), p->pos() );
}
int iuse::handle_ground_graffiti( player &p, item *it, const std::string &prefix,
const tripoint &pt )
{
std::string message = string_input_popup()
.title( prefix + " " + _( "(To delete, input one '.')" ) )
.identifier( "graffiti" )
.query_string();
if( message.empty() ) {
return 0;
} else {
const auto where = pt != p.pos() ? pt : p.pos();
bool grave = g->m.ter( where ) == t_grave_new;
int move_cost;
if( message == "." ) {
if( g->m.has_graffiti_at( where ) ) {
move_cost = 3 * g->m.graffiti_at( where ).length();
g->m.delete_graffiti( where );
if( grave ) {
add_msg( _( "You blur the inscription on the grave." ) );
} else {
add_msg( _( "You manage to get rid of the message on the ground." ) );
}
} else {
add_msg( _( "There isn't anything to erase here." ) );
return 0;
}
} else {
g->m.set_graffiti( where, message );
if( grave ) {
add_msg( _( "You carve an inscription on the grave." ) );
} else {
add_msg( _( "You write a message on the ground." ) );
}
move_cost = 2 * message.length();
}
p.moves -= move_cost;
}
if( it != nullptr ) {
return it->type->charges_to_use();
} else {
return 0;
}
}
/**
* Heats up a food item.
* @return 1 if an item was heated, false if nothing was heated.
*/
static bool heat_item( player &p )
{
auto loc = g->inv_map_splice( []( const item & itm ) {
return( ( itm.is_food() && !itm.item_tags.count( "HOT" ) ) ||
( itm.is_food_container() && !itm.contents.front().item_tags.count( "HOT" ) ) );
}, _( "Heat up what?" ), 1, _( "You don't have appropriate food to heat up." ) );
item *heat = loc.get_item();
if( heat == nullptr ) {
add_msg( m_info, _( "Never mind." ) );
return false;
}
item &target = heat->is_food_container() ? heat->contents.front() : *heat;
p.mod_moves( -to_turns<int>( 3_seconds ) ); //initial preparations
// simulates heat capacity of food, more weight = longer heating time
// this is x2 to simulate larger delta temperature of frozen food in relation to
// heating non-frozen food (x1); no real life physics here, only aproximations
int move_mod = to_turns<int>( time_duration::from_seconds( to_gram( target.weight() ) ) );
if( target.item_tags.count( "FROZEN" ) ) {
target.apply_freezerburn();
if( target.has_flag( "EATEN_COLD" ) ) {
target.cold_up();
add_msg( _( "You defrost the food, but don't heat it up, since you enjoy it cold." ) );
} else {
add_msg( _( "You defrost and heat up the food." ) );
target.heat_up();
// x2 because we have to defrost and heat
move_mod *= 2;
}
} else {
add_msg( _( "You heat up the food." ) );
target.heat_up();
}
p.mod_moves( -move_mod ); // time needed to actually heat up
return true;
}
int iuse::heatpack( player *p, item *it, bool, const tripoint & )
{
if( heat_item( *p ) ) {
it->convert( "heatpack_used" );
}
return 0;
}
int iuse::heat_food( player *p, item *it, bool, const tripoint & )
{
if( g->m.has_nearby_fire( p->pos() ) ) {
heat_item( *p );
} else if( p->has_active_bionic( bionic_id( "bio_tools" ) ) && p->power_level > 10 &&
query_yn( _( "There is no fire around, use your integrated toolset instead?" ) ) ) {
if( heat_item( *p ) ) {
p->charge_power( -10 );
}
} else {
p->add_msg_if_player( m_info, _( "You need to be next to fire to heat something up with the %s." ),
it->tname() );
}
return 0;
}
int iuse::hotplate( player *p, item *it, bool, const tripoint & )
{
if( it->typeId() != "atomic_coffeepot" && ( !it->units_sufficient( *p ) ) ) {
p->add_msg_if_player( m_info, _( "The %s's batteries are dead." ), it->tname() );
return 0;
}
int choice = 0;
if( ( p->has_effect( effect_bite ) || p->has_effect( effect_bleed ) ||
p->has_trait( trait_MASOCHIST ) ||
p->has_trait( trait_MASOCHIST_MED ) || p->has_trait( trait_CENOBITE ) ) && !p->is_underwater() ) {
//Might want to cauterize
choice = uilist( _( "Using hotplate:" ), {
_( "Heat food" ), _( "Cauterize wound" )
} );
}
if( choice == 0 ) {
if( heat_item( *p ) ) {
return it->type->charges_to_use();
}
} else if( choice == 1 ) {
return cauterize_elec( *p, *it );
}
return 0;
}
int iuse::towel( player *p, item *it, bool t, const tripoint & )
{
if( t ) {
// Continuous usage, do nothing as not initiated by the player, this is for
// wet towels only as they are active items.
return 0;
}
bool slime = p->has_effect( effect_slimed );
bool boom = p->has_effect( effect_boomered );
bool glow = p->has_effect( effect_glowing );
int mult = slime + boom + glow; // cleaning off more than one at once makes it take longer
bool towelUsed = false;
// can't use an already wet towel!
if( it->has_flag( "WET" ) ) {
p->add_msg_if_player( m_info, _( "That %s is too wet to soak up any more liquid!" ),
it->tname() );
// clean off the messes first, more important
} else if( slime || boom || glow ) {
p->remove_effect( effect_slimed ); // able to clean off all at once
p->remove_effect( effect_boomered );
p->remove_effect( effect_glowing );
p->add_msg_if_player( _( "You use the %s to clean yourself off, saturating it with slime!" ),
it->tname() );
towelUsed = true;
if( it->typeId() == "towel" ) {
it->convert( "towel_soiled" );
}
// dry off from being wet
} else if( abs( p->has_morale( MORALE_WET ) ) ) {
p->rem_morale( MORALE_WET );
p->body_wetness.fill( 0 );
p->add_msg_if_player( _( "You use the %s to dry off, saturating it with water!" ),
it->tname() );
towelUsed = true;
it->item_counter = to_turns<int>( 30_minutes );
// default message
} else {
p->add_msg_if_player( _( "You are already dry, the %s does nothing." ), it->tname() );
}
// towel was used
if( towelUsed ) {
if( mult == 0 ) {
mult = 1;
}
p->moves -= 50 * mult;
// change "towel" to a "towel_wet" (different flavor text/color)
if( it->typeId() == "towel" ) {
it->convert( "towel_wet" );
}
// WET, active items have their timer decremented every turn
it->item_tags.insert( "WET" );
it->active = true;
}
return it->type->charges_to_use();
}
int iuse::unfold_generic( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
vehicle *veh = g->m.add_vehicle( vproto_id( "none" ), p->posx(), p->posy(), 0, 0, 0, false );
if( veh == nullptr ) {
p->add_msg_if_player( m_info, _( "There's no room to unfold the %s." ), it->tname() );
return 0;
}
veh->name = it->get_var( "vehicle_name" );
if( !veh->restore( it->get_var( "folding_bicycle_parts" ) ) ) {
g->m.destroy_vehicle( veh );
return 0;
}
const bool can_float = size( veh->get_avail_parts( "FLOATS" ) ) > 2;
const auto invalid_pos = []( const tripoint & pp, bool can_float ) {
return ( g->m.has_flag_ter( TFLAG_DEEP_WATER, pp ) && !can_float ) ||
g->m.veh_at( pp ) || g->m.impassable( pp );
};
for( const vpart_reference &vp : veh->get_all_parts() ) {
if( vp.info().location != "STRUCTURE" ) {
continue;
}
const tripoint pp = vp.pos();
if( invalid_pos( pp, can_float ) ) {
p->add_msg_if_player( m_info, _( "There's no room to unfold the %s." ), it->tname() );
g->m.destroy_vehicle( veh );
return 0;
}
}
g->m.add_vehicle_to_cache( veh );
std::string unfold_msg = it->get_var( "unfold_msg" );
if( unfold_msg.empty() ) {
unfold_msg = _( "You painstakingly unfold the %s and make it ready to ride." );
} else {
unfold_msg = _( unfold_msg );
}
faction *yours = g->faction_manager_ptr->get( faction_id( "your_followers" ) );
veh->set_owner( yours );
p->add_msg_if_player( m_neutral, unfold_msg, veh->name );
p->moves -= it->get_var( "moves", to_turns<int>( 5_seconds ) );
return 1;
}
int iuse::adrenaline_injector( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() && p->get_effect_dur( effect_adrenaline ) >= 30_minutes ) {
return 0;
}
p->moves -= to_turns<int>( 1_seconds );
p->add_msg_player_or_npc( _( "You inject yourself with adrenaline." ),
_( "<npcname> injects themselves with adrenaline." ) );
item syringe( "syringe", it->birthday() );
p->i_add( syringe );
if( p->has_effect( effect_adrenaline ) ) {
p->add_msg_if_player( m_bad, _( "Your heart spasms!" ) );
// Note: not the mod, the health
p->mod_healthy( -20 );
}
p->add_effect( effect_adrenaline, 20_minutes );
return it->type->charges_to_use();
}
int iuse::jet_injector( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
p->add_msg_if_player( m_info, _( "The jet injector is empty." ) );
return 0;
} else {
p->add_msg_if_player( _( "You inject yourself with the jet injector." ) );
// Intensity is 2 here because intensity = 1 is the comedown
p->add_effect( effect_jetinjector, 20_minutes, num_bp, false, 2 );
p->mod_painkiller( 20 );
p->stim += 10;
p->healall( 20 );
}
if( p->has_effect( effect_jetinjector ) ) {
if( p->get_effect_dur( effect_jetinjector ) > 20_minutes ) {
p->add_msg_if_player( m_warning, _( "Your heart is beating alarmingly fast!" ) );
}
}
return it->type->charges_to_use();
}
int iuse::stimpack( player *p, item *it, bool, const tripoint & )
{
if( p->get_item_position( it ) >= -1 ) {
p->add_msg_if_player( m_info,
_( "You must wear the stimulant delivery system before you can activate it." ) );
return 0;
}
if( !it->ammo_sufficient() ) {
p->add_msg_if_player( m_info, _( "The stimulant delivery system is empty." ) );
return 0;
} else {
p->add_msg_if_player( _( "You inject yourself with the stimulants." ) );
// Intensity is 2 here because intensity = 1 is the comedown
p->add_effect( effect_stimpack, 25_minutes, num_bp, false, 2 );
p->mod_painkiller( 2 );
p->stim += 20;
p->mod_fatigue( -100 );
p->stamina = p->get_stamina_max();
}
return it->type->charges_to_use();
}
int iuse::radglove( player *p, item *it, bool, const tripoint & )
{
if( p->get_item_position( it ) >= -1 ) {
p->add_msg_if_player( m_info,
_( "You must wear the radiation biomonitor before you can activate it." ) );
return 0;
} else if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_info, _( "The radiation biomonitor needs batteries to function." ) );
return 0;
} else {
p->add_msg_if_player( _( "You activate your radiation biomonitor." ) );
if( p->radiation >= 1 ) {
p->add_msg_if_player( m_warning, _( "You are currently irradiated." ) );
p->add_msg_player_or_say( m_info,
_( "Your radiation level: %d mSv." ),
_( "It says here that my radiation level is %d mSv." ),
p->radiation );
} else {
p->add_msg_player_or_say( m_info,
_( "You are not currently irradiated." ),
_( "It says I'm not irradiated" ) );
}
p->add_msg_if_player( _( "Have a nice day!" ) );
}
return it->type->charges_to_use();
}
int iuse::contacts( player *p, item *it, bool, const tripoint & )
{
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
const time_duration duration = rng( 6_days, 8_days );
if( p->has_effect( effect_contacts ) ) {
if( query_yn( _( "Replace your current lenses?" ) ) ) {
p->moves -= to_turns<int>( 20_seconds );
p->add_msg_if_player( _( "You replace your current %s." ), it->tname() );
p->remove_effect( effect_contacts );
p->add_effect( effect_contacts, duration );
return it->type->charges_to_use();
} else {
p->add_msg_if_player( _( "You don't do anything with your %s." ), it->tname() );
return 0;
}
} else if( p->has_trait( trait_HYPEROPIC ) || p->has_trait( trait_MYOPIC ) ||
p->has_trait( trait_URSINE_EYE ) ) {
p->moves -= to_turns<int>( 20_seconds );
p->add_msg_if_player( _( "You put the %s in your eyes." ), it->tname() );
p->add_effect( effect_contacts, duration );
return it->type->charges_to_use();
} else {
p->add_msg_if_player( m_info, _( "Your vision is fine already." ) );
return 0;
}
}
int iuse::talking_doll( player *p, item *it, bool, const tripoint & )
{
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_info, _( "The %s's batteries are dead." ), it->tname() );
return 0;
}
const SpeechBubble speech = get_speech( it->typeId() );
sounds::sound( p->pos(), speech.volume, sounds::sound_t::speech, speech.text, true, "speech",
it->typeId() );
// Sound code doesn't describe noises at the player position
if( p->can_hear( p->pos(), speech.volume ) ) {
p->add_msg_if_player( _( "You hear \"%s\"" ), speech.text );
}
return it->type->charges_to_use();
}
int iuse::gun_repair( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
/** @EFFECT_MECHANICS >1 allows gun repair */
if( p->get_skill_level( skill_mechanics ) < 2 ) {
p->add_msg_if_player( m_info, _( "You need a mechanics skill of 2 to use this repair kit." ) );
return 0;
}
int inventory_index = g->inv_for_all( _( "Select the firearm to repair" ) );
item &fix = p->i_at( inventory_index );
if( fix.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return 0;
}
if( !fix.is_firearm() ) {
p->add_msg_if_player( m_info, _( "That isn't a firearm!" ) );
return 0;
}
if( fix.has_flag( "NO_REPAIR" ) ) {
p->add_msg_if_player( m_info, _( "You cannot repair your %s." ), fix.tname() );
return 0;
}
if( fix.damage() <= fix.min_damage() ) {
p->add_msg_if_player( m_info, _( "You cannot improve your %s any more this way." ),
fix.tname() );
return 0;
}
if( fix.damage() <= 0 && p->get_skill_level( skill_mechanics ) < 8 ) {
p->add_msg_if_player( m_info, _( "Your %s is already in peak condition." ), fix.tname() );
p->add_msg_if_player( m_info,
_( "With a higher mechanics skill, you might be able to improve it." ) );
return 0;
}
/** @EFFECT_MECHANICS >=8 allows accurizing ranged weapons */
const std::string startdurability = fix.durability_indicator( true );
std::string resultdurability;
if( fix.damage() <= 0 ) {
sounds::sound( p->pos(), 6, sounds::sound_t::activity, "crunch", true, "tool", "repair_kit" );
p->moves -= to_turns<int>( 20_seconds * p->fine_detail_vision_mod() );
p->practice( skill_mechanics, 10 );
fix.mod_damage( -itype::damage_scale );
p->add_msg_if_player( m_good, _( "You accurize your %s." ), fix.tname( 1, false ) );
} else if( fix.damage() > itype::damage_scale ) {
sounds::sound( p->pos(), 8, sounds::sound_t::activity, "crunch", true, "tool", "repair_kit" );
p->moves -= to_turns<int>( 10_seconds * p->fine_detail_vision_mod() );
p->practice( skill_mechanics, 10 );
fix.mod_damage( -itype::damage_scale );
resultdurability = fix.durability_indicator( true );
p->add_msg_if_player( m_good, _( "You repair your %s! ( %s-> %s)" ), fix.tname( 1, false ),
startdurability, resultdurability );
} else {
sounds::sound( p->pos(), 8, sounds::sound_t::activity, "crunch", true, "tool", "repair_kit" );
p->moves -= to_turns<int>( 5_seconds * p->fine_detail_vision_mod() );
p->practice( skill_mechanics, 10 );
fix.set_damage( 0 );
resultdurability = fix.durability_indicator( true );
p->add_msg_if_player( m_good, _( "You repair your %s completely! ( %s-> %s)" ),
fix.tname( 1, false ), startdurability, resultdurability );
}
return it->type->charges_to_use();
}
int iuse::gunmod_attach( player *p, item *it, bool, const tripoint & )
{
if( !it || !it->is_gunmod() ) {
debugmsg( "tried to attach non-gunmod" );
return 0;
}
if( !p ) {
return 0;
}
auto loc = game_menus::inv::gun_to_modify( *p, *it );
if( !loc ) {
add_msg( m_info, _( "Never mind." ) );
return 0;
}
p->gunmod_add( *loc, *it );
return 0;
}
int iuse::toolmod_attach( player *p, item *it, bool, const tripoint & )
{
if( !it || !it->is_toolmod() ) {
debugmsg( "tried to attach non-toolmod" );
return 0;
}
if( !p ) {
return 0;
}
auto filter = [&it]( const item & e ) {
// don't allow ups battery mods on a UPS or UPS-powered tools
if( it->has_flag( "USE_UPS" ) && ( e.typeId() == "UPS_off" || e.typeId() == "adv_UPS_off" ||
e.has_flag( "USE_UPS" ) ) ) {
return false;
}
// can only attach to unmodified tools that use compatible ammo
return e.is_tool() && e.toolmods().empty() && !e.magazine_current() &&
std::any_of( it->type->mod->acceptable_ammo.begin(),
it->type->mod->acceptable_ammo.end(), [&]( const ammotype & at ) {
return e.ammo_types( false ).count( at );
} );
};
auto loc = g->inv_map_splice( filter, _( "Select tool to modify" ), 1,
_( "You don't have compatible tools." ) );
if( !loc ) {
add_msg( m_info, _( "Never mind." ) );
return 0;
}
if( loc->ammo_remaining() ) {
if( !g->unload( *loc ) ) {
p->add_msg_if_player( m_info, _( "You cancel unloading the tool." ) );
return 0;
}
}
p->toolmod_add( std::move( loc ), item_location( *p, it ) );
return 0;
}
int iuse::misc_repair( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
return 0;
}
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( p->fine_detail_vision_mod() > 4 ) {
add_msg( m_info, _( "You can't see to repair!" ) );
return 0;
}
/** @EFFECT_FABRICATION >0 allows use of repair kit */
if( p->get_skill_level( skill_fabrication ) < 1 ) {
p->add_msg_if_player( m_info, _( "You need a fabrication skill of 1 to use this repair kit." ) );
return 0;
}
static const std::set<material_id> repairable {
material_id( "wood" ),
material_id( "paper" ),
material_id( "bone" ),
material_id( "chitin" ),
material_id( "acidchitin" )
};
int inventory_index = g->inv_for_filter( _( "Select the item to repair" ), []( const item & itm ) {
return !itm.is_firearm() && itm.made_of_any( repairable ) && !itm.count_by_charges();
} );
item &fix = p->i_at( inventory_index );
if( fix.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return 0;
}
if( fix.damage() <= fix.min_damage() ) {
p->add_msg_if_player( m_info, _( "You cannot improve your %s any more this way." ),
fix.tname() );
return 0;
}
if( fix.damage() <= 0 && fix.has_flag( "PRIMITIVE_RANGED_WEAPON" ) ) {
p->add_msg_if_player( m_info, _( "You cannot improve your %s any more this way." ),
fix.tname() );
return 0;
}
const std::string startdurability = fix.durability_indicator( true );
std::string resultdurability;
if( fix.damage() <= 0 ) {
p->moves -= to_turns<int>( 10_seconds * p->fine_detail_vision_mod() );
p->practice( skill_fabrication, 10 );
fix.mod_damage( -itype::damage_scale );
p->add_msg_if_player( m_good, _( "You reinforce your %s." ), fix.tname() );
} else if( fix.damage() > itype::damage_scale ) {
p->moves -= to_turns<int>( 5_seconds * p->fine_detail_vision_mod() );
p->practice( skill_fabrication, 10 );
fix.mod_damage( -itype::damage_scale );
resultdurability = fix.durability_indicator( true );
p->add_msg_if_player( m_good, _( "You repair your %s! ( %s-> %s)" ), fix.tname( 1, false ),
startdurability, resultdurability );
} else {
p->moves -= to_turns<int>( 3_seconds * p->fine_detail_vision_mod() );
p->practice( skill_fabrication, 10 );
fix.set_damage( 0 );
resultdurability = fix.durability_indicator( true );
p->add_msg_if_player( m_good, _( "You repair your %s completely! ( %s-> %s)" ),
fix.tname( 1, false ), startdurability, resultdurability );
}
return it->type->charges_to_use();
}
int iuse::bell( player *p, item *it, bool, const tripoint & )
{
if( it->typeId() == "cow_bell" ) {
sounds::sound( p->pos(), 12, sounds::sound_t::music, _( "Clank! Clank!" ), true, "misc",
"cow_bell" );
if( !p->is_deaf() ) {
const int cow_factor = 1 + ( p->mutation_category_level.find( "CATTLE" ) ==
p->mutation_category_level.end() ?
0 :
( p->mutation_category_level.find( "CATTLE" )->second ) / 8
);
if( x_in_y( cow_factor, 1 + cow_factor ) ) {
p->add_morale( MORALE_MUSIC, 1, 15 * ( cow_factor > 10 ? 10 : cow_factor ) );
}
}
} else {
sounds::sound( p->pos(), 4, sounds::sound_t::music, _( "Ring! Ring!" ), true, "misc", "bell" );
}
return it->type->charges_to_use();
}
int iuse::seed( player *p, item *it, bool, const tripoint & )
{
if( p->is_npc() ||
query_yn( _( "Sure you want to eat the %s? You could plant it in a mound of dirt." ),
colorize( it->tname(), it->color_in_inventory() ) ) ) {
return it->type->charges_to_use(); //This eats the seed object.
}
return 0;
}
bool iuse::robotcontrol_can_target( player *p, const monster &m )
{
return !m.is_dead()
&& m.type->in_species( ROBOT )
&& m.friendly == 0
&& rl_dist( p->pos(), m.pos() ) <= 10;
}
int iuse::robotcontrol( player *p, item *it, bool, const tripoint & )
{
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( _( "The %s's batteries are dead." ), it->tname() );
return 0;
}
if( p->has_trait( trait_ILLITERATE ) ) {
p->add_msg_if_player( _( "You cannot read a computer screen." ) );
return 0;
}
if( p->has_trait( trait_HYPEROPIC ) && !p->worn_with_flag( "FIX_FARSIGHT" ) &&
!p->has_effect( effect_contacts ) && !p->has_bionic( bionic_id( "bio_eye_optic" ) ) ) {
add_msg( m_info, _( "You'll need to put on reading glasses before you can see the screen." ) );
return 0;
}
int choice = uilist( _( "Welcome to hackPRO!:" ), {
_( "Prepare IFF protocol override" ),
_( "Set friendly robots to passive mode" ),
_( "Set friendly robots to combat mode" )
} );
switch( choice ) {
case 0: { // attempt to make a robot friendly
uilist pick_robot;
pick_robot.text = _( "Choose an endpoint to hack." );
// Build a list of all unfriendly robots in range.
std::vector< std::shared_ptr< monster> > mons; // TODO: change into vector<Creature*>
std::vector< tripoint > locations;
int entry_num = 0;
for( const monster &candidate : g->all_monsters() ) {
if( robotcontrol_can_target( p, candidate ) ) {
mons.push_back( g->shared_from( candidate ) );
pick_robot.addentry( entry_num++, true, MENU_AUTOASSIGN, candidate.name() );
tripoint seen_loc;
// Show locations of seen robots, center on player if robot is not seen
if( p->sees( candidate ) ) {
seen_loc = candidate.pos();
} else {
seen_loc = p->pos();
}
locations.push_back( seen_loc );
}
}
if( mons.empty() ) {
p->add_msg_if_player( m_info, _( "No enemy robots in range." ) );
return it->type->charges_to_use();
}
pointmenu_cb callback( locations );
pick_robot.callback = &callback;
pick_robot.query();
if( pick_robot.ret < 0 || static_cast<size_t>( pick_robot.ret ) >= mons.size() ) {
p->add_msg_if_player( m_info, _( "Never mind" ) );
return it->type->charges_to_use();
}
const size_t mondex = pick_robot.ret;
std::shared_ptr< monster > z = mons[mondex];
p->add_msg_if_player( _( "You start reprogramming the %s into an ally." ), z->name() );
/** @EFFECT_INT speeds up hacking preperation */
/** @EFFECT_COMPUTER speeds up hacking preperation */
int move_cost = std::max( 100, 1000 - p->int_cur * 10 - p->get_skill_level( skill_computer ) * 10 );
player_activity act( activity_id( "ACT_ROBOT_CONTROL" ), move_cost );
act.monsters.emplace_back( z );
p->assign_activity( act );
return it->type->charges_to_use();
}
case 1: { //make all friendly robots stop their purposeless extermination of (un)life.
p->moves -= to_turns<int>( 1_seconds );
int f = 0; //flag to check if you have robotic allies
for( monster &critter : g->all_monsters() ) {
if( critter.friendly != 0 && critter.type->in_species( ROBOT ) ) {
p->add_msg_if_player( _( "A following %s goes into passive mode." ),
critter.name() );
critter.add_effect( effect_docile, 1_turns, num_bp, true );
f = 1;
}
}
if( f == 0 ) {
p->add_msg_if_player( _( "You are not commanding any robots." ) );
return 0;
}
return it->type->charges_to_use();
}
case 2: { //make all friendly robots terminate (un)life with extreme prejudice
p->moves -= to_turns<int>( 1_seconds );
int f = 0; //flag to check if you have robotic allies
for( monster &critter : g->all_monsters() ) {
if( critter.friendly != 0 && critter.has_flag( MF_ELECTRONIC ) ) {
p->add_msg_if_player( _( "A following %s goes into combat mode." ),
critter.name() );
critter.remove_effect( effect_docile );
f = 1;
}
}
if( f == 0 ) {
p->add_msg_if_player( _( "You are not commanding any robots." ) );
return 0;
}
return it->type->charges_to_use();
}
}
return 0;
}
static void init_memory_card_with_random_stuff( item &it )
{
if( it.has_flag( "MC_MOBILE" ) && ( it.has_flag( "MC_RANDOM_STUFF" ) ||
it.has_flag( "MC_SCIENCE_STUFF" ) ) && !( it.has_flag( "MC_USED" ) ||
it.has_flag( "MC_HAS_DATA" ) ) ) {
it.item_tags.insert( "MC_HAS_DATA" );
bool encrypted = false;
if( it.has_flag( "MC_MAY_BE_ENCRYPTED" ) && one_in( 8 ) ) {
it.convert( it.typeId() + "_encrypted" );
}
//some special cards can contain "MC_ENCRYPTED" flag
if( it.has_flag( "MC_ENCRYPTED" ) ) {
encrypted = true;
}
int data_chance = 2;
//encrypted memory cards often contain data
if( encrypted && !one_in( 3 ) ) {
data_chance--;
}
//just empty memory card
if( !one_in( data_chance ) ) {
return;
}
//add someone's personal photos
if( one_in( data_chance ) ) {
//decrease chance to more data
data_chance++;
if( encrypted && one_in( 3 ) ) {
data_chance--;
}
const int duckfaces_count = rng( 5, 30 );
it.set_var( "MC_PHOTOS", duckfaces_count );
}
//decrease chance to music and other useful data
data_chance++;
if( encrypted && one_in( 2 ) ) {
data_chance--;
}
if( one_in( data_chance ) ) {
data_chance++;
if( encrypted && one_in( 3 ) ) {
data_chance--;
}
const int new_songs_count = rng( 5, 15 );
it.set_var( "MC_MUSIC", new_songs_count );
}
data_chance++;
if( encrypted && one_in( 2 ) ) {
data_chance--;
}
if( one_in( data_chance ) ) {
it.set_var( "MC_RECIPE", "SIMPLE" );
}
if( it.has_flag( "MC_SCIENCE_STUFF" ) ) {
it.set_var( "MC_RECIPE", "SCIENCE" );
}
}
}
static bool einkpc_download_memory_card( player &p, item &eink, item &mc )
{
bool something_downloaded = false;
if( mc.get_var( "MC_PHOTOS", 0 ) > 0 ) {
something_downloaded = true;
int new_photos = mc.get_var( "MC_PHOTOS", 0 );
mc.erase_var( "MC_PHOTOS" );
p.add_msg_if_player( m_good, ngettext( "You download %d new photo into internal memory.",
"You download %d new photos into internal memory.", new_photos ), new_photos );
const int old_photos = eink.get_var( "EIPC_PHOTOS", 0 );
eink.set_var( "EIPC_PHOTOS", old_photos + new_photos );
}
if( mc.get_var( "MC_MUSIC", 0 ) > 0 ) {
something_downloaded = true;
int new_songs = mc.get_var( "MC_MUSIC", 0 );
mc.erase_var( "MC_MUSIC" );
p.add_msg_if_player( m_good, ngettext( "You download %d new song into internal memory.",
"You download %d new songs into internal memory.", new_songs ), new_songs );
const int old_songs = eink.get_var( "EIPC_MUSIC", 0 );
eink.set_var( "EIPC_MUSIC", old_songs + new_songs );
}
if( !mc.get_var( "MC_RECIPE" ).empty() ) {
const bool science = mc.get_var( "MC_RECIPE" ) == "SCIENCE";
mc.erase_var( "MC_RECIPE" );
std::vector<const recipe *> candidates;
for( const auto &e : recipe_dict ) {
const auto &r = e.second;
if( science ) {
if( r.difficulty >= 3 && one_in( r.difficulty + 1 ) ) {
candidates.push_back( &r );
}
} else {
if( r.category == "CC_FOOD" ) {
if( r.difficulty <= 3 && one_in( r.difficulty ) ) {
candidates.push_back( &r );
}
}
}
}
if( !candidates.empty() ) {
const recipe *r = random_entry( candidates );
const recipe_id &rident = r->ident();
const auto old_recipes = eink.get_var( "EIPC_RECIPES" );
if( old_recipes.empty() ) {
something_downloaded = true;
eink.set_var( "EIPC_RECIPES", "," + rident.str() + "," );
p.add_msg_if_player( m_good, _( "You download a recipe for %s into the tablet's memory." ),
r->result_name() );
} else {
if( old_recipes.find( "," + rident.str() + "," ) == std::string::npos ) {
something_downloaded = true;
eink.set_var( "EIPC_RECIPES", old_recipes + rident.str() + "," );
p.add_msg_if_player( m_good, _( "You download a recipe for %s into the tablet's memory." ),
r->result_name() );
} else {
p.add_msg_if_player( m_good, _( "Your tablet already has a recipe for %s." ),
r->result_name() );
}
}
}
}
if( mc.has_var( "MC_EXTENDED_PHOTOS" ) ) {
std::vector<extended_photo_def> extended_photos;
try {
item_read_extended_photos( mc, extended_photos, "MC_EXTENDED_PHOTOS" );
item_read_extended_photos( eink, extended_photos, "EIPC_EXTENDED_PHOTOS", true );
item_write_extended_photos( eink, extended_photos, "EIPC_EXTENDED_PHOTOS" );
something_downloaded = true;
p.add_msg_if_player( m_good, _( "You have downloaded your photos." ) );
} catch( const JsonError &e ) {
debugmsg( "Error card reading photos (loaded photos = %i) : %s", extended_photos.size(),
e.c_str() );
}
}
const auto monster_photos = mc.get_var( "MC_MONSTER_PHOTOS" );
if( !monster_photos.empty() ) {
something_downloaded = true;
p.add_msg_if_player( m_good, _( "You have updated your monster collection." ) );
auto photos = eink.get_var( "EINK_MONSTER_PHOTOS" );
if( photos.empty() ) {
eink.set_var( "EINK_MONSTER_PHOTOS", monster_photos );
} else {
std::istringstream f( monster_photos );
std::string s;
while( getline( f, s, ',' ) ) {
if( s.empty() ) {
continue;
}
const std::string mtype = s;
getline( f, s, ',' );
char *chq = &s[0];
const int quality = atoi( chq );
const size_t eink_strpos = photos.find( "," + mtype + "," );
if( eink_strpos == std::string::npos ) {
photos += mtype + "," + string_format( "%d", quality ) + ",";
} else {
const size_t strqpos = eink_strpos + mtype.size() + 2;
char *chq = &photos[strqpos];
const int old_quality = atoi( chq );
if( quality > old_quality ) {
chq = &string_format( "%d", quality )[0];
photos[strqpos] = *chq;
}
}
}
eink.set_var( "EINK_MONSTER_PHOTOS", photos );
}
}
if( mc.has_flag( "MC_TURN_USED" ) ) {
mc.clear_vars();
mc.unset_flags();
mc.convert( "mobile_memory_card_used" );
}
if( !something_downloaded ) {
p.add_msg_if_player( m_info, _( "This memory card does not contain any new data." ) );
return false;
}
return true;
}
static std::string photo_quality_name( const int index )
{
static const std::array<std::string, 6> names {
{
//~ photo quality adjective
{ translate_marker( "awful" ) }, { translate_marker( "bad" ) }, { translate_marker( "not bad" ) }, { translate_marker( "good" ) }, { translate_marker( "fine" ) }, { translate_marker( "exceptional" ) }
}
};
return _( names[index] );
}
int iuse::einktabletpc( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) {
if( !it->get_var( "EIPC_MUSIC_ON" ).empty() && ( it->ammo_remaining() > 0 ) ) {
if( calendar::once_every( 5_minutes ) ) {
it->ammo_consume( 1, p->pos() );
}
//the more varied music, the better max mood.
const int songs = it->get_var( "EIPC_MUSIC", 0 );
play_music( *p, pos, 8, std::min( 25, songs ) );
} else {
it->active = false;
it->erase_var( "EIPC_MUSIC_ON" );
p->add_msg_if_player( m_info, _( "Tablet's batteries are dead." ) );
}
return 0;
} else if( !p->is_npc() ) {
enum {
ei_invalid, ei_photo, ei_music, ei_recipe, ei_uploaded_photos, ei_monsters, ei_download, ei_decrypt
};
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( p->has_trait( trait_ILLITERATE ) ) {
add_msg( m_info, _( "You cannot read a computer screen." ) );
return 0;
}
if( p->has_trait( trait_HYPEROPIC ) && !p->worn_with_flag( "FIX_FARSIGHT" ) &&
!p->has_effect( effect_contacts ) && !p->has_bionic( bionic_id( "bio_eye_optic" ) ) ) {
add_msg( m_info, _( "You'll need to put on reading glasses before you can see the screen." ) );
return 0;
}
uilist amenu;
amenu.text = _( "Choose menu option:" );
const int photos = it->get_var( "EIPC_PHOTOS", 0 );
if( photos > 0 ) {
amenu.addentry( ei_photo, true, 'p', _( "Unsorted photos [%d]" ), photos );
} else {
amenu.addentry( ei_photo, false, 'p', _( "No photos on device" ) );
}
const int songs = it->get_var( "EIPC_MUSIC", 0 );
if( songs > 0 ) {
if( it->active ) {
amenu.addentry( ei_music, true, 'm', _( "Turn music off" ) );
} else {
amenu.addentry( ei_music, true, 'm', _( "Turn music on [%d]" ), songs );
}
} else {
amenu.addentry( ei_music, false, 'm', _( "No music on device" ) );
}
if( !it->get_var( "EIPC_RECIPES" ).empty() ) {
amenu.addentry( ei_recipe, true, 'r', _( "View recipes on E-ink screen" ) );
}
if( !it->get_var( "EIPC_EXTENDED_PHOTOS" ).empty() ) {
amenu.addentry( ei_uploaded_photos, true, 'l', _( "Your photos" ) );
}
if( !it->get_var( "EINK_MONSTER_PHOTOS" ).empty() ) {
amenu.addentry( ei_monsters, true, 'y', _( "Your collection of monsters" ) );
} else {
amenu.addentry( ei_monsters, false, 'y', _( "Collection of monsters is empty" ) );
}
amenu.addentry( ei_download, true, 'w', _( "Download data from memory card" ) );
/** @EFFECT_COMPUTER >2 allows decrypting memory cards more easily */
if( p->get_skill_level( skill_computer ) > 2 ) {
amenu.addentry( ei_decrypt, true, 'd', _( "Decrypt memory card" ) );
} else {
amenu.addentry( ei_decrypt, false, 'd', _( "Decrypt memory card (low skill)" ) );
}
amenu.query();
const int choice = amenu.ret;
if( ei_photo == choice ) {
const int photos = it->get_var( "EIPC_PHOTOS", 0 );
const int viewed = std::min( photos, static_cast<int>( rng( 10, 30 ) ) );
const int count = photos - viewed;
if( count == 0 ) {
it->erase_var( "EIPC_PHOTOS" );
} else {
it->set_var( "EIPC_PHOTOS", count );
}
p->moves -= to_turns<int>( rng( 3_seconds, 7_seconds ) );
if( p->has_trait( trait_PSYCHOPATH ) ) {
p->add_msg_if_player( m_info, _( "Wasted time, these pictures do not provoke your senses." ) );
} else {
p->add_morale( MORALE_PHOTOS, rng( 15, 30 ), 100 );
const int random_photo = rng( 1, 20 );
switch( random_photo ) {
case 1:
p->add_msg_if_player( m_good, _( "You used to have a dog like this..." ) );
break;
case 2:
p->add_msg_if_player( m_good, _( "Ha-ha! An amusing cat photo." ) );
break;
case 3:
p->add_msg_if_player( m_good, _( "Excellent pictures of nature." ) );
break;
case 4:
p->add_msg_if_player( m_good, _( "Food photos... your stomach rumbles!" ) );
break;
case 5:
p->add_msg_if_player( m_good, _( "Some very interesting travel photos." ) );
break;
case 6:
p->add_msg_if_player( m_good, _( "Pictures of a concert of popular band." ) );
break;
case 7:
p->add_msg_if_player( m_good, _( "Photos of someone's luxurious house." ) );
break;
default:
p->add_msg_if_player( m_good, _( "You feel nostalgic as you stare at the photo." ) );
break;
}
}
return it->type->charges_to_use();
}
if( ei_music == choice ) {
p->moves -= 30;
if( it->active ) {
it->active = false;
it->erase_var( "EIPC_MUSIC_ON" );
p->add_msg_if_player( m_info, _( "You turned off music on your %s." ), it->tname() );
} else {
it->active = true;
it->set_var( "EIPC_MUSIC_ON", "1" );
p->add_msg_if_player( m_info, _( "You turned on music on your %s." ), it->tname() );
}
return it->type->charges_to_use();
}
if( ei_recipe == choice ) {
p->moves -= 50;
uilist rmenu;
rmenu.text = _( "List recipes:" );
std::vector<recipe_id> candidate_recipes;
std::istringstream f( it->get_var( "EIPC_RECIPES" ) );
std::string s;
int k = 0;
while( getline( f, s, ',' ) ) {
if( s.empty() ) {
continue;
}
candidate_recipes.emplace_back( s );
const auto &recipe = *candidate_recipes.back();
if( recipe ) {
rmenu.addentry( k++, true, -1, recipe.result_name() );
}
}
rmenu.query();
return it->type->charges_to_use();
}
if( ei_uploaded_photos == choice ) {
show_photo_selection( *p, *it, "EIPC_EXTENDED_PHOTOS" );
return it->type->charges_to_use();
}
if( ei_monsters == choice ) {
uilist pmenu;
pmenu.text = _( "Your collection of monsters:" );
std::vector<mtype_id> monster_photos;
std::istringstream f( it->get_var( "EINK_MONSTER_PHOTOS" ) );
std::string s;
int k = 0;
while( getline( f, s, ',' ) ) {
if( s.empty() ) {
continue;
}
monster_photos.push_back( mtype_id( s ) );
std::string menu_str;
const monster dummy( monster_photos.back() );
menu_str = dummy.name();
getline( f, s, ',' );
char *chq = &s[0];
const int quality = atoi( chq );
menu_str += " [" + photo_quality_name( quality ) + "]";
pmenu.addentry( k++, true, -1, menu_str.c_str() );
}
int choice;
do {
pmenu.query();
choice = pmenu.ret;
if( choice < 0 ) {
break;
}
const monster dummy( monster_photos[choice] );
popup( dummy.type->get_description() );
} while( true );
return it->type->charges_to_use();
}
if( ei_download == choice ) {
p->moves -= to_turns<int>( 2_seconds );
const int inventory_index = g->inv_for_flag( "MC_MOBILE", _( "Insert memory card" ) );
item &mc = p->i_at( inventory_index );
if( mc.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return it->type->charges_to_use();
}
if( !mc.has_flag( "MC_MOBILE" ) ) {
p->add_msg_if_player( m_info, _( "This is not a compatible memory card." ) );
return it->type->charges_to_use();
}
init_memory_card_with_random_stuff( mc );
if( mc.has_flag( "MC_ENCRYPTED" ) ) {
p->add_msg_if_player( m_info, _( "This memory card is encrypted." ) );
return it->type->charges_to_use();
}
if( !mc.has_flag( "MC_HAS_DATA" ) ) {
p->add_msg_if_player( m_info, _( "This memory card does not contain any new data." ) );
return it->type->charges_to_use();
}
einkpc_download_memory_card( *p, *it, mc );
return it->type->charges_to_use();
}
if( ei_decrypt == choice ) {
p->moves -= to_turns<int>( 2_seconds );
const int inventory_index = g->inv_for_flag( "MC_MOBILE", _( "Insert memory card" ) );
item &mc = p->i_at( inventory_index );
if( mc.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return it->type->charges_to_use();
}
if( !mc.has_flag( "MC_MOBILE" ) ) {
p->add_msg_if_player( m_info, _( "This is not a compatible memory card." ) );
return it->type->charges_to_use();
}
init_memory_card_with_random_stuff( mc );
if( !mc.has_flag( "MC_ENCRYPTED" ) ) {
p->add_msg_if_player( m_info, _( "This memory card is not encrypted." ) );
return it->type->charges_to_use();
}
p->practice( skill_computer, rng( 2, 5 ) );
/** @EFFECT_INT increases chance of safely decrypting memory card */
/** @EFFECT_COMPUTER increases chance of safely decrypting memory card */
const int success = p->get_skill_level( skill_computer ) * rng( 1,
p->get_skill_level( skill_computer ) ) *
rng( 1, p->int_cur ) - rng( 30, 80 );
if( success > 0 ) {
p->practice( skill_computer, rng( 5, 10 ) );
p->add_msg_if_player( m_good, _( "You successfully decrypted content on %s!" ),
mc.tname() );
einkpc_download_memory_card( *p, *it, mc );
} else {
if( success > -10 || one_in( 5 ) ) {
p->add_msg_if_player( m_neutral, _( "You failed to decrypt the %s." ), mc.tname() );
} else {
p->add_msg_if_player( m_bad,
_( "You tripped the firmware protection, and the card deleted its data!" ) );
mc.clear_vars();
mc.unset_flags();
mc.convert( "mobile_memory_card_used" );
}
}
return it->type->charges_to_use();
}
}
return 0;
}
struct extended_photo_def : public JsonDeserializer, public JsonSerializer {
int quality;
std::string name;
std::string description;
extended_photo_def() = default;
void deserialize( JsonIn &jsin ) override {
JsonObject obj = jsin.get_object();
quality = obj.get_int( "quality" );
name = obj.get_string( "name" );
description = obj.get_string( "description" );
}
void serialize( JsonOut &jsout ) const override {
jsout.start_object();
jsout.member( "quality", quality );
jsout.member( "name", name );
jsout.member( "description", description );
jsout.end_object();
}
};
static std::string colorized_trap_name_at( const tripoint &point )
{
const trap &trap = g->m.tr_at( point );
std::string name;
if( !trap.is_null() && trap.get_visibility() <= 1 ) {
name = colorize( trap.name(), trap.color ) + _( " on " );
}
return name;
}
static std::string colorized_field_description_at( const tripoint &point )
{
static const std::unordered_set<field_type_id, std::hash<int>> covered_in_affix_ids = {
fd_blood, fd_bile, fd_gibs_flesh, fd_gibs_veggy, fd_web,
fd_slime, fd_acid, fd_sap, fd_sludge, fd_blood_veggy,
fd_blood_insect, fd_blood_invertebrate, fd_gibs_insect,
fd_gibs_invertebrate, fd_rubble
};
static const std::unordered_set<field_type_id, std::hash<int>> on_affix_ids = {
fd_fire, fd_flame_burst
};
static const std::unordered_set<field_type_id, std::hash<int>> under_affix_ids = {
fd_gas_vent, fd_fire_vent, fd_fatigue
};
static const std::unordered_set<field_type_id, std::hash<int>> illuminated_by_affix_ids = {
fd_spotlight, fd_laser, fd_dazzling, fd_electricity
};
static const std::vector<std::pair<std::unordered_set<field_type_id, std::hash<int>>, std::string>>
affixes_vec = {
{ covered_in_affix_ids, _( " covered in %s" ) },
{ on_affix_ids, _( " on %s" ) },
{ under_affix_ids, _( " under %s" ) },
{ illuminated_by_affix_ids, _( " illuminated by %s" ) }
}; // anything else is "in %s cloud"
std::string field_text;
const field &field = g->m.field_at( point );
const field_entry *entry = field.find_field( field.displayed_field_type() );
if( entry ) {
std::string affix;
for( const auto &pair : affixes_vec ) {
if( pair.first.find( entry->get_field_type() ) != pair.first.end() ) {
affix = pair.second;
}
}
if( affix.empty() ) {
field_text = string_format( " in %s cloud", colorize( entry->name(), entry->color() ) );
} else {
field_text = string_format( affix, colorize( entry->name(), entry->color() ) );
}
}
return field_text;
}
static std::string colorized_item_name( const item &item )
{
nc_color color = item.color_in_inventory();
std::string damtext = item.damage() != 0 ? item.durability_indicator() : "";
return damtext + colorize( item.tname( 1, false ), color );
}
static std::string colorized_item_description( const item &item )
{
std::vector<iteminfo> dummy;
iteminfo_query query = iteminfo_query(
std::vector<iteminfo_parts> {
iteminfo_parts::DESCRIPTION,
iteminfo_parts::DESCRIPTION_NOTES,
iteminfo_parts::DESCRIPTION_CONTENTS
} ) ;
return item.info( dummy, &query, 1 );
}
static const item get_top_item_at_point( const tripoint &point,
const units::volume min_visible_volume )
{
map_stack items = g->m.i_at( point );
// iterate from topmost item down to ground
for( const item &it : items ) {
if( it.volume() > min_visible_volume ) {
// return top (or first big enough) item to the list
return it;
}
}
return item();
}
static std::string colorized_ter_name_flags_at( const tripoint &point,
const std::vector<std::string> &flags, const std::vector<ter_str_id> &ter_whitelist )
{
const ter_id ter = g->m.ter( point );
std::string name = colorize( ter->name(), ter->color() );
const std::string &graffiti_message = g->m.graffiti_at( point );
if( !graffiti_message.empty() ) {
name += string_format( _( " with graffiti \"%s\"" ), graffiti_message );
return name;
}
if( ter_whitelist.empty() && flags.empty() ) {
return name;
}
if( !ter->open.is_null() || ( ter->examine != iexamine::none &&
ter->examine != iexamine::fungus &&
ter->examine != iexamine::water_source &&
ter->examine != iexamine::dirtmound ) ) {
return name;
}
for( const ter_str_id &ter_good : ter_whitelist ) {
if( ter->id == ter_good ) {
return name;
}
}
for( const std::string &flag : flags ) {
if( ter->has_flag( flag ) ) {
return name;
}
}
return std::string();
}
static std::string colorized_feature_description_at( const tripoint &center_point, bool &item_found,
const units::volume min_visible_volume )
{
item_found = false;
const furn_id furn = g->m.furn( center_point );
if( furn != f_null && furn.is_valid() ) {
std::string furn_str = colorize( furn->name(), c_yellow );
std::string sign_message = g->m.get_signage( center_point );
if( !sign_message.empty() ) {
furn_str += string_format( _( " with message \"%s\"" ), sign_message );
}
if( !furn->has_flag( "CONTAINER" ) && !furn->has_flag( "SEALED" ) ) {
const item item = get_top_item_at_point( center_point, min_visible_volume );
if( !item.is_null() ) {
furn_str += string_format( _( " with %s on it" ), colorized_item_name( item ) );
item_found = true;
}
}
return furn_str;
}
return std::string();
}
static std::string format_object_pair( const std::pair<std::string, int> &pair,
const std::string &article )
{
if( pair.second == 1 ) {
return string_format( "%s%s", article, pair.first );
} else if( pair.second > 1 ) {
return string_format( "%s%i %s", article, pair.second, pair.first );
}
return std::string();
}
static std::string format_object_pair_article( const std::pair<std::string, int> &pair )
{
return format_object_pair( pair, pgettext( "Article 'a', replace it with empty "
"string if it is not used in language", "a " ) );
}
static std::string format_object_pair_no_article( const std::pair<std::string, int> &pair )
{
return format_object_pair( pair, "" );
}
static std::string effects_description_for_creature( Creature *const creature, std::string &pose,
const std::string &pronoun_sex )
{
struct ef_con { // effect constraint
std::string status;
std::string pose;
int intensity_lower_limit;
ef_con( std::string status, std::string pose, int intensity_lower_limit ) :
status( status ), pose( pose ), intensity_lower_limit( intensity_lower_limit ) {}
ef_con( std::string status, std::string pose ) :
status( status ), pose( pose ), intensity_lower_limit( 0 ) {}
ef_con( std::string status, int intensity_lower_limit ) :
status( status ), pose(), intensity_lower_limit( intensity_lower_limit ) {}
ef_con( std::string status ) :
status( status ), pose(), intensity_lower_limit( 0 ) {}
};
static const std::unordered_map<efftype_id, ef_con> vec_effect_status = {
{ effect_onfire, ef_con( _( " is on <color_red>fire</color>. " ) ) },
{ effect_bleed, ef_con( _( " is <color_red>bleeding</color>. " ), 1 ) },
{ effect_happy, ef_con( _( " looks <color_green>happy</color>. " ), 13 ) },
{ effect_downed, ef_con( "", _( "downed" ) ) },
{ effect_in_pit, ef_con( "", _( "stuck" ) ) },
{ effect_stunned, ef_con( _( " is <color_blue>stunned</color>. " ) ) },
{ effect_dazed, ef_con( _( " is <color_blue>dazed</color>. " ) ) },
{ effect_beartrap, ef_con( _( " is stuck in beartrap. " ) ) },
{ effect_laserlocked, ef_con( _( " have tiny <color_red>red dot</color> on body. " ) ) },
{ effect_boomered, ef_con( _( " is covered in <color_magenta>bile</color>. " ) ) },
{ effect_glowing, ef_con( _( " is covered in <color_yellow>glowing goo</color>. " ) ) },
{ effect_slimed, ef_con( _( " is covered in <color_green>thick goo</color>. " ) ) },
{ effect_corroding, ef_con( _( " is covered in <color_light_green>acid</color>. " ) ) },
{ effect_sap, ef_con( _( " is coated in <color_brown>sap</color>. " ) ) },
{ effect_webbed, ef_con( _( " is covered in <color_gray>webs</color>. " ) ) },
{ effect_spores, ef_con( _( " is covered in <color_green>spores</color>. " ), 1 ) },
{ effect_crushed, ef_con( _( " lies under <color_gray>collapsed debris</color>. " ), _( "lies" ) ) },
{ effect_lack_sleep, ef_con( _( " looks <color_gray>very tired</color>. " ) ) },
{ effect_lying_down, ef_con( _( " is <color_dark_blue>sleeping</color>. " ), _( "lies" ) ) },
{ effect_sleep, ef_con( _( " is <color_dark_blue>sleeping</color>. " ), _( "lies" ) ) },
{ effect_haslight, ef_con( _( " is <color_yellow>lit</color>. " ) ) },
{ effect_saddled, ef_con( _( " is <color_gray>saddled</color>. " ) ) },
{ effect_harnessed, ef_con( _( " is being <color_gray>harnessed</color> by a vehicle. " ) ) },
{ effect_monster_armor, ef_con( _( " is <color_gray>wearing armor</color>. " ) ) },
{ effect_has_bag, ef_con( _( " have <color_gray>bag</color> attached. " ) ) },
{ effect_tied, ef_con( _( " is <color_gray>tied</color>. " ) ) },
{ effect_bouldering, ef_con( "", _( "balancing" ) ) }
};
std::string figure_effects;
if( creature ) {
for( const auto &pair : vec_effect_status ) {
if( creature->get_effect_int( pair.first ) > pair.second.intensity_lower_limit ) {
if( !pair.second.status.empty() ) {
figure_effects += pronoun_sex + pair.second.status;
}
if( !pair.second.pose.empty() ) {
pose = pair.second.pose;
}
}
}
if( creature->has_effect( effect_sad ) ) {
int intensity = creature->get_effect_int( effect_sad );
if( intensity > 500 && intensity <= 950 ) {
figure_effects += pronoun_sex + pgettext( "Someone", " looks <color_blue>sad</color>. " );
} else if( intensity > 950 ) {
figure_effects += pronoun_sex + pgettext( "Someone", " looks <color_blue>depressed</color>. " );
}
}
float pain = creature->get_pain() / 10.f;
if( pain > 3 ) {
figure_effects += pronoun_sex + pgettext( "Someone", " is writhing in <color_red>pain</color>. " );
}
if( creature->has_effect( effect_riding ) ) {
pose = _( "rides" );
monster *const mon = g->critter_at<monster>( creature->pos(), false );
figure_effects += pronoun_sex + string_format( _( " is riding %s. " ),
colorize( mon->name(), c_light_blue ) );
}
if( creature->has_effect( effect_glowing_led ) ) {
figure_effects += _( "A bionic LED is <color_yellow>glowing</color> softly. " );
}
}
if( !figure_effects.empty() ) { // remove last space
figure_effects.erase( figure_effects.end() - 1 );
}
return figure_effects;
}
struct object_names_collection {
std::unordered_map<std::string, int>
furniture,
vehicles,
items,
terrain;
std::string figure_text;
std::string obj_nearby_text;
};
static object_names_collection enumerate_objects_around_point( const tripoint &point,
const int radius, const tripoint &bounds_center_point, const int bounds_radius,
const tripoint &camera_pos, const units::volume min_visible_volume, bool create_figure_desc,
std::unordered_set<tripoint> &ignored_points,
std::unordered_set<const vehicle *> &vehicles_recorded )
{
const tripoint_range bounds = g->m.points_in_radius( bounds_center_point, bounds_radius );
const tripoint_range points_in_radius = g->m.points_in_radius( point, radius );
int dist = rl_dist( camera_pos, point );
bool item_found = false;
std::unordered_set<const vehicle *> local_vehicles_recorded( vehicles_recorded );
object_names_collection ret_obj;
std::string description_part_on_figure;
std::string description_furniture_on_figure;
std::string description_terrain_on_figure;
// store objects in radius
for( const tripoint &point_around_figure : points_in_radius ) {
if( !bounds.is_point_inside( point_around_figure ) ||
!g->m.sees( camera_pos, point_around_figure, dist + radius ) ||
( ignored_points.find( point_around_figure ) != ignored_points.end() &&
!( point_around_figure == point && create_figure_desc ) ) ) {
continue; // disallow photos with not visible objects
}
units::volume volume_to_search = point_around_figure == bounds_center_point ? 0_ml :
min_visible_volume;
std::string furn_desc = colorized_feature_description_at( point_around_figure, item_found,
volume_to_search );
const item item = get_top_item_at_point( point_around_figure, volume_to_search );
const optional_vpart_position veh_part_pos = g->m.veh_at( point_around_figure );
std::string unusual_ter_desc = colorized_ter_name_flags_at( point_around_figure,
camera_ter_whitelist_flags,
camera_ter_whitelist_types );
std::string ter_desc = colorized_ter_name_flags_at( point_around_figure );
const std::string trap_name = colorized_trap_name_at( point_around_figure );
const std::string field_desc = colorized_field_description_at( point_around_figure );
if( !furn_desc.empty() ) {
furn_desc = trap_name + furn_desc + field_desc;
if( point == point_around_figure && create_figure_desc ) {
description_furniture_on_figure = furn_desc;
} else {
ret_obj.furniture[ furn_desc ] ++;
}
} else if( veh_part_pos.has_value() ) {
const vehicle veh = veh_part_pos->vehicle();
const std::string veh_name = colorize( veh.disp_name(), c_light_blue );
const vehicle *veh_hash = &veh_part_pos->vehicle();
if( local_vehicles_recorded.find( veh_hash ) == local_vehicles_recorded.end() &&
point != point_around_figure ) {
// new vehicle, point is not center
ret_obj.vehicles[ veh_name ] ++;
} else if( point == point_around_figure ) {
// point is center
description_part_on_figure = string_format( _( "%1$s from %2$s" ),
veh_part_pos.part_displayed()->part().name(), veh_name );
if( ret_obj.vehicles.find( veh_name ) != ret_obj.vehicles.end() &&
local_vehicles_recorded.find( veh_hash ) != local_vehicles_recorded.end() ) {
// remove vehicle name only if we previously added THIS vehicle name (in case of same name)
ret_obj.vehicles[ veh_name ] --;
if( ret_obj.vehicles[ veh_name ] <= 0 ) {
ret_obj.vehicles.erase( veh_name );
}
}
}
vehicles_recorded.insert( veh_hash );
local_vehicles_recorded.insert( veh_hash );
} else if( !item.is_null() ) {
std::string item_name = colorized_item_name( item );
item_name = trap_name + item_name + field_desc;
if( point == point_around_figure && create_figure_desc ) {
description_terrain_on_figure = string_format( _( "%1$s with a %2$s" ), ter_desc, item_name );
} else {
ret_obj.items[ item_name ] ++;
}
} else if( !unusual_ter_desc.empty() ) {
unusual_ter_desc = trap_name + unusual_ter_desc + field_desc;
if( point == point_around_figure && create_figure_desc ) {
description_furniture_on_figure = unusual_ter_desc;
} else {
ret_obj.furniture[ unusual_ter_desc ] ++;
}
} else if( !ter_desc.empty() && ( !field_desc.empty() || !trap_name.empty() ) ) {
ter_desc = trap_name + ter_desc + field_desc;
if( point == point_around_figure && create_figure_desc ) {
description_terrain_on_figure = ter_desc;
} else {
ret_obj.terrain[ ter_desc ] ++;
}
} else {
ter_desc = trap_name + ter_desc + field_desc;
if( point == point_around_figure && create_figure_desc ) {
description_terrain_on_figure = ter_desc;
}
}
ignored_points.insert( point_around_figure );
}
if( create_figure_desc ) {
std::vector<std::string> objects_combined_desc;
std::unordered_map<std::string, int> vecs_to_retrieve[4] = {
ret_obj.furniture, ret_obj.vehicles, ret_obj.items, ret_obj.terrain
};
for( int i = 0; i < 4; i++ ) {
for( const auto &p : vecs_to_retrieve[ i ] ) {
objects_combined_desc.push_back( i == 1 ? // vehicle name already includes "the"
format_object_pair_no_article( p ) : format_object_pair_article( p ) );
}
}
const char *transl_str = pgettext( "someone stands/sits *on* something", " on a %s." );
if( !description_part_on_figure.empty() ) {
ret_obj.figure_text = string_format( transl_str, description_part_on_figure );
} else {
if( !description_furniture_on_figure.empty() ) {
ret_obj.figure_text = string_format( transl_str, description_furniture_on_figure );
} else {
ret_obj.figure_text = string_format( transl_str, description_terrain_on_figure );
}
}
if( !objects_combined_desc.empty() ) {
// store objects to description_figures_status
std::string objects_text = enumerate_as_string( objects_combined_desc );
ret_obj.obj_nearby_text = string_format( ngettext( "Nearby is %s.", "Nearby are %s.",
objects_combined_desc.size() ), objects_text );
}
}
return ret_obj;
}
static extended_photo_def photo_def_for_camera_point( const tripoint &aim_point,
const tripoint &camera_pos,
std::vector<monster *> &monster_vec, std::vector<player *> &player_vec )
{
// look for big items on top of stacks in the background for the selfie description
const units::volume min_visible_volume = 490_ml;
std::unordered_set<tripoint> ignored_points;
std::unordered_set<const vehicle *> vehicles_recorded;
std::unordered_map<std::string, std::string> description_figures_appearance;
std::vector<std::pair<std::string, std::string>> description_figures_status;
std::string timestamp = to_string( time_point( calendar::turn ) );
int dist = rl_dist( camera_pos, aim_point );
const tripoint_range bounds = g->m.points_in_radius( aim_point, 2 );
extended_photo_def photo;
bool need_store_weather = false;
int outside_tiles_num = 0;
int total_tiles_num = 0;
const auto map_deincrement_or_erase = []( std::unordered_map<std::string, int> &obj_map,
const std::string & key ) {
if( obj_map.find( key ) != obj_map.end() ) {
obj_map[ key ] --;
if( obj_map[ key ] <= 0 ) {
obj_map.erase( key );
}
}
};
// first scan for critters and mark nearby furniture, vehicles and items
for( const tripoint &current : bounds ) {
if( !g->m.sees( camera_pos, current, dist + 3 ) ) {
continue; // disallow photos with non-visible objects
}
monster *const mon = g->critter_at<monster>( current, false );
player *guy = g->critter_at<player>( current );
total_tiles_num++;
if( g->m.is_outside( current ) ) {
need_store_weather = true;
outside_tiles_num++;
}
if( guy || mon ) {
std::string figure_appearance, figure_name, pose, pronoun_sex, figure_effects;
Creature *creature;
if( mon && mon->has_effect( effect_ridden ) ) {
// only player can ride, see monexamine::mount_pet
guy = &g->u;
description_figures_appearance[ mon->name() ] = "\"" + mon->type->get_description() + "\"";
}
if( guy ) {
if( guy->is_hallucination() ) {
continue; // do not include hallucinations
}
if( guy->movement_mode_is( PMM_CROUCH ) ) {
pose = _( "sits" );
} else {
pose = _( "stands" );
}
const std::vector<std::string> vec = guy->short_description_parts();
figure_appearance = join( vec, "\n\n" );
figure_name = guy->name;
pronoun_sex = guy->male ? _( "He" ) : _( "She" );
creature = guy;
player_vec.push_back( guy );
} else {
if( mon->is_hallucination() || mon->type->in_species( HALLUCINATION ) ) {
continue; // do not include hallucinations
}
pose = _( "stands" );
figure_appearance = "\"" + mon->type->get_description() + "\"";
figure_name = mon->name();
pronoun_sex = pgettext( "Pronoun", "It" );
creature = mon;
monster_vec.push_back( mon );
}
figure_effects = effects_description_for_creature( creature, pose, pronoun_sex );
description_figures_appearance[ figure_name ] = figure_appearance;
object_names_collection obj_collection = enumerate_objects_around_point( current, 1, aim_point, 2,
camera_pos, min_visible_volume, true,
ignored_points, vehicles_recorded );
std::string figure_text = pose + obj_collection.figure_text;
if( !figure_effects.empty() ) {
figure_text += " " + figure_effects;
}
if( !obj_collection.obj_nearby_text.empty() ) {
figure_text += " " + obj_collection.obj_nearby_text;
}
auto name_text_pair = std::pair<std::string, std::string>( figure_name, figure_text );
if( current == aim_point ) {
description_figures_status.insert( description_figures_status.begin(), name_text_pair );
} else {
description_figures_status.push_back( name_text_pair );
}
}
}
// scan for everythin NOT near critters
object_names_collection obj_coll = enumerate_objects_around_point( aim_point, 2, aim_point, 2,
camera_pos, min_visible_volume, false,
ignored_points, vehicles_recorded );
std::string photo_text = _( "This is a photo of " );
bool found_item_aim_point;
std::string furn_desc = colorized_feature_description_at( aim_point, found_item_aim_point,
0_ml ) ;
const item item = get_top_item_at_point( aim_point, 0_ml );
const std::string trap_name = colorized_trap_name_at( aim_point );
std::string ter_name = colorized_ter_name_flags_at( aim_point, {}, {} );
const std::string field_desc = colorized_field_description_at( aim_point );
bool found_vehicle_aim_point = g->m.veh_at( aim_point ).has_value(),
found_furniture_aim_point = !furn_desc.empty();
// colorized_feature_description_at do not update flag if no furniture found, so need to check again
if( !found_furniture_aim_point ) {
found_item_aim_point = !item.is_null();
}
const ter_id ter_aim = g->m.ter( aim_point );
const furn_id furn_aim = g->m.furn( aim_point );
if( !description_figures_status.empty() ) {
std::string names = enumerate_as_string( description_figures_status.begin(),
description_figures_status.end(),
[]( const std::pair<std::string, std::string> &it ) {
return colorize( it.first, c_light_blue );
} );
photo.name = names;
photo_text += names + ".";
for( const auto &figure_status : description_figures_status ) {
photo_text += "\n\n" + colorize( figure_status.first, c_light_blue )
+ " " + figure_status.second;
}
} else if( found_vehicle_aim_point ) {
const optional_vpart_position veh_part_pos = g->m.veh_at( aim_point );
const std::string veh_name = colorize( veh_part_pos->vehicle().disp_name(), c_light_blue );
photo.name = veh_name;
photo_text += veh_name + ".";
map_deincrement_or_erase( obj_coll.vehicles, veh_name );
} else if( found_furniture_aim_point || found_item_aim_point ) {
std::string item_name = colorized_item_name( item );
if( found_furniture_aim_point ) {
furn_desc = trap_name + furn_desc + field_desc;
photo.name = furn_desc;
photo_text += photo.name + ".";
map_deincrement_or_erase( obj_coll.furniture, furn_desc );
} else if( found_item_aim_point ) {
item_name = trap_name + item_name + field_desc;
photo.name = item_name;
photo_text += item_name + ". " + string_format( _( "It lies on the %s." ),
ter_name );
map_deincrement_or_erase( obj_coll.items, item_name );
}
if( found_furniture_aim_point && !furn_aim->description.empty() ) {
photo_text += "\n\n" + colorize( furn_aim->name(), c_yellow ) + ":\n" + furn_aim->description;
}
if( found_item_aim_point ) {
photo_text += "\n\n" + item_name + ":\n" + colorized_item_description( item );
}
} else {
ter_name = trap_name + ter_name + field_desc;
photo.name = ter_name;
photo_text += photo.name + ".";
map_deincrement_or_erase( obj_coll.terrain, ter_name );
map_deincrement_or_erase( obj_coll.furniture, ter_name );
if( !ter_aim->description.empty() ) {
photo_text += "\n\n" + photo.name + ":\n" + ter_aim->description;
}
}
if( !obj_coll.items.empty() ) {
std::string obj_list = enumerate_as_string( obj_coll.items.begin(), obj_coll.items.end(),
format_object_pair_article );
photo_text += "\n\n" + string_format( ngettext( "There is something lying on the ground: %s.",
"There are some things lying on the ground: %s.", obj_coll.items.size() ),
obj_list );
}
if( !obj_coll.furniture.empty() ) {
std::string obj_list = enumerate_as_string( obj_coll.furniture.begin(), obj_coll.furniture.end(),
format_object_pair_article );
photo_text += "\n\n" + string_format( ngettext( "Something is visible in the background: %s.",
"Some objects are visible in the background: %s.", obj_coll.furniture.size() ),
obj_list );
}
if( !obj_coll.vehicles.empty() ) {
std::string obj_list = enumerate_as_string( obj_coll.vehicles.begin(), obj_coll.vehicles.end(),
format_object_pair_no_article );
photo_text += "\n\n" + string_format( ngettext( "There is %s parked in the background.",
"There are %s parked in the background.", obj_coll.vehicles.size() ),
obj_list );
}
if( !obj_coll.terrain.empty() ) {
std::string obj_list = enumerate_as_string( obj_coll.terrain.begin(), obj_coll.terrain.end(),
format_object_pair_article );
photo_text += "\n\n" + string_format( ngettext( "There is %s in the background.",
"There are %s in the background.", obj_coll.terrain.size() ),
obj_list );
}
const oter_id &cur_ter = overmap_buffer.ter( ms_to_omt_copy( g->m.getabs( aim_point ) ) );
std::string overmap_desc = string_format( _( "In the background you can see a %s" ),
colorize( cur_ter->get_name(), cur_ter->get_color() ) );
if( outside_tiles_num == total_tiles_num ) {
photo_text += _( "\n\nThis photo was taken <color_gray>outside</color>." );
} else if( outside_tiles_num == 0 ) {
photo_text += _( "\n\nThis photo was taken <color_gray>inside</color>." );
overmap_desc += _( " interior" );
} else if( outside_tiles_num < total_tiles_num / 2.0 ) {
photo_text += _( "\n\nThis photo was taken mostly <color_gray>inside</color>,"
" but <color_gray>outside</color> can be seen." );
overmap_desc += _( " interior" );
} else if( outside_tiles_num >= total_tiles_num / 2.0 ) {
photo_text += _( "\n\nThis photo was taken mostly <color_gray>outside</color>,"
" but <color_gray>inside</color> can be seen." );
}
photo_text += "\n" + overmap_desc + ".";
if( g->get_levz() >= 0 && need_store_weather ) {
photo_text += "\n\n";
if( calendar::turn.is_sunrise_now() ) {
photo_text += _( "It is <color_yellow>sunrise</color>. " );
} else if( calendar::turn.is_sunset_now() ) {
photo_text += _( "It is <color_light_red>sunset</color>. " );
} else if( calendar::turn.is_night() ) {
photo_text += _( "It is <color_dark_gray>night</color>. " );
} else {
photo_text += _( "It is day. " );
}
const weather_datum w_data = weather_data( g->weather.weather );
photo_text += string_format( _( "The weather is %s." ), colorize( w_data.name, w_data.color ) );
}
for( const auto &figure : description_figures_appearance ) {
photo_text += "\n\n" + string_format( _( "%s appearance:" ),
colorize( figure.first, c_light_blue ) ) + "\n" + figure.second;
}
photo_text += "\n\n" + string_format( pgettext( "Date", "The photo was taken on %s." ),
colorize( timestamp, c_light_blue ) );
photo.description = photo_text;
return photo;
}
static void item_save_monsters( player &p, item &it, const std::vector<monster *> &monster_vec,
const int photo_quality )
{
std::string monster_photos = it.get_var( "CAMERA_MONSTER_PHOTOS" );
if( monster_photos.empty() ) {
monster_photos = ",";
}
for( monster * const &monster_p : monster_vec ) {
const std::string mtype = monster_p->type->id.str();
const std::string name = monster_p->name();
// position of <monster type string>
const size_t mon_str_pos = monster_photos.find( "," + mtype + "," );
if( mon_str_pos == std::string::npos ) { // new monster
monster_photos += string_format( "%s,%d,", mtype, photo_quality );
} else { // replace quality character, if new photo is better
const size_t quality_num_pos = mon_str_pos + mtype.size() + 2;
char *quality_char = &monster_photos[ quality_num_pos ];
const int old_quality = atoi( quality_char ); // get qual number from char
if( photo_quality > old_quality ) {
monster_photos[ quality_num_pos ] = string_format( "%d", photo_quality )[ 0 ];
}
if( !p.is_blind() ) {
if( photo_quality > old_quality ) {
p.add_msg_if_player( m_good, _( "The quality of %s image is better than the previous one." ),
colorize( name, c_light_blue ) );
} else if( old_quality == 5 ) {
p.add_msg_if_player( _( "The quality of stored %s image is already maximally detailed." ),
colorize( name, c_light_blue ) );
} else {
p.add_msg_if_player( m_bad, _( "But the quality of %s image is worse than the previous one." ),
colorize( name, c_light_blue ) );
}
}
}
}
it.set_var( "CAMERA_MONSTER_PHOTOS", monster_photos );
}
// throws exception
static bool item_read_extended_photos( item &it, std::vector<extended_photo_def> &extended_photos,
const std::string &var_name, bool insert_at_begin )
{
bool result = false;
std::istringstream extended_photos_data( it.get_var( var_name ) );
JsonIn json( extended_photos_data );
if( insert_at_begin ) {
std::vector<extended_photo_def> temp_vec;
result = json.read( temp_vec );
extended_photos.insert( std::begin( extended_photos ), std::begin( temp_vec ),
std::end( temp_vec ) );
} else {
result = json.read( extended_photos );
}
return result;
}
// throws exception
static void item_write_extended_photos( item &it,
const std::vector<extended_photo_def> &extended_photos,
const std::string &var_name )
{
std::ostringstream extended_photos_data;
JsonOut json( extended_photos_data );
json.write( extended_photos );
it.set_var( var_name, extended_photos_data.str() );
}
static bool show_photo_selection( player &p, item &it, const std::string &var_name )
{
if( p.is_blind() ) {
p.add_msg_if_player( _( "You can't see the camera screen, you're blind." ) );
return false;
}
uilist pmenu;
pmenu.text = _( "Photos saved on camera:" );
std::vector<std::string> descriptions;
std::vector<extended_photo_def> extended_photos;
try {
item_read_extended_photos( it, extended_photos, var_name );
} catch( const JsonError &e ) {
debugmsg( "Error reading photos: %s", e.c_str() );
}
try { // if there is old photos format, append them; delete old and save new
if( item_read_extended_photos( it, extended_photos, "CAMERA_NPC_PHOTOS", true ) ) {
it.erase_var( "CAMERA_NPC_PHOTOS" );
item_write_extended_photos( it, extended_photos, var_name );
}
} catch( const JsonError &e ) {
debugmsg( "Error migrating old photo format: %s", e.c_str() );
}
int k = 0;
for( const extended_photo_def &extended_photo : extended_photos ) {
std::string menu_str = extended_photo.name;
size_t index = menu_str.find( p.name );
if( index != std::string::npos ) {
menu_str.replace( index, p.name.length(), "You" );
}
descriptions.push_back( extended_photo.description );
menu_str += " [" + photo_quality_name( extended_photo.quality ) + "]";
pmenu.addentry( k++, true, -1, menu_str.c_str() );
}
int choice;
do {
pmenu.query();
choice = pmenu.ret;
if( choice < 0 ) {
break;
}
popup( "%s", descriptions[choice].c_str() );
} while( true );
return true;
}
int iuse::camera( player *p, item *it, bool, const tripoint & )
{
enum {c_shot, c_photos, c_monsters, c_upload};
// CAMERA_NPC_PHOTOS is old save variable
bool found_extended_photos = !it->get_var( "CAMERA_NPC_PHOTOS" ).empty() ||
!it->get_var( "CAMERA_EXTENDED_PHOTOS" ).empty();
bool found_monster_photos = !it->get_var( "CAMERA_MONSTER_PHOTOS" ).empty();
uilist amenu;
amenu.text = _( "What to do with camera?" );
amenu.addentry( c_shot, true, 't', _( "Take a photo" ) );
if( !found_extended_photos && !found_monster_photos ) {
amenu.addentry( c_photos, false, 'l', _( "No photos in memory" ) );
} else {
if( found_extended_photos ) {
amenu.addentry( c_photos, true, 'l', _( "List photos" ) );
}
if( found_monster_photos ) {
amenu.addentry( c_monsters, true, 'm', _( "Your collection of monsters" ) );
}
amenu.addentry( c_upload, true, 'u', _( "Upload photos to memory card" ) );
}
amenu.query();
const int choice = amenu.ret;
if( choice < 0 ) {
return 0;
}
if( c_shot == choice ) {
const cata::optional<tripoint> aim_point_ = g->look_around();
if( !aim_point_ ) {
p->add_msg_if_player( _( "Never mind." ) );
return 0;
}
tripoint aim_point = *aim_point_;
bool incorrect_focus = false;
tripoint_range aim_bounds = g->m.points_in_radius( aim_point, 2 );
std::vector<tripoint> trajectory = line_to( p->pos(), aim_point, 0, 0 );
trajectory.push_back( aim_point );
p->moves -= 50;
sounds::sound( p->pos(), 8, sounds::sound_t::activity, _( "Click." ), true, "tool",
"camera_shutter" );
for( std::vector<tripoint>::iterator point_it = trajectory.begin();
point_it != trajectory.end();
++point_it ) {
const tripoint trajectory_point = *point_it;
if( point_it != trajectory.end() ) {
const tripoint next_point = *( point_it + 1 ); // Trajectory ends on last visible tile
if( !g->m.sees( p->pos(), next_point, rl_dist( p->pos(), next_point ) + 3 ) ) {
p->add_msg_if_player( _( "You have the wrong camera focus." ) );
incorrect_focus = true;
// recalculate target point
aim_point = trajectory_point;
aim_bounds = g->m.points_in_radius( trajectory_point, 2 );
}
}
monster *const mon = g->critter_at<monster>( trajectory_point, true );
player *const guy = g->critter_at<player>( trajectory_point );
if( mon || guy || trajectory_point == aim_point ) {
int dist = rl_dist( p->pos(), trajectory_point );
int camera_bonus = it->has_flag( "CAMERA_PRO" ) ? 10 : 0;
int photo_quality = 20 - rng( dist, dist * 2 ) * 2 + rng( camera_bonus / 2, camera_bonus );
if( photo_quality > 5 ) {
photo_quality = 5;
}
if( photo_quality < 0 ) {
photo_quality = 0;
}
if( p->is_blind() ) {
photo_quality /= 2;
}
if( mon ) {
monster &z = *mon;
// shoot past small monsters and hallucinations
if( trajectory_point != aim_point && ( z.type->size <= MS_SMALL || z.is_hallucination() ||
z.type->in_species( HALLUCINATION ) ) ) {
continue;
}
if( !aim_bounds.is_point_inside( trajectory_point ) ) {
// take a photo of the monster that's in the way
p->add_msg_if_player( m_warning, _( "A %s got in the way of your photo." ), z.name() );
incorrect_focus = true;
} else if( trajectory_point != aim_point ) { // shoot past mon that will be in photo anyway
continue;
}
// get an special message if the target is a hallucination
if( trajectory_point == aim_point && ( z.is_hallucination() ||
z.type->in_species( HALLUCINATION ) ) ) {
p->add_msg_if_player( _( "Strange... there's nothing in the center of picture?" ) );
}
} else if( guy ) {
if( trajectory_point == aim_point && guy->is_hallucination() ) {
p->add_msg_if_player( _( "Strange... %s's not visible on the picture?" ), guy->name );
} else if( !aim_bounds.is_point_inside( trajectory_point ) ) {
// take a photo of the monster that's in the way
p->add_msg_if_player( m_warning, _( "%s got in the way of your photo." ), guy->name );
incorrect_focus = true;
} else if( trajectory_point != aim_point ) { // shoot past guy that will be in photo anyway
continue;
}
}
if( incorrect_focus ) {
photo_quality = photo_quality == 0 ? 0 : photo_quality - 1;
}
std::vector<extended_photo_def> extended_photos;
std::vector<monster *> monster_vec;
std::vector<player *> player_vec;
extended_photo_def photo = photo_def_for_camera_point( trajectory_point, p->pos(), monster_vec,
player_vec );
photo.quality = photo_quality;
try {
item_read_extended_photos( *it, extended_photos, "CAMERA_EXTENDED_PHOTOS" );
extended_photos.push_back( photo );
item_write_extended_photos( *it, extended_photos, "CAMERA_EXTENDED_PHOTOS" );
} catch( const JsonError &e ) {
debugmsg( "Error when adding new photo (loaded photos = %i): %s", extended_photos.size(),
e.c_str() );
}
const bool selfie = std::find( player_vec.begin(), player_vec.end(), p ) != player_vec.end();
if( selfie ) {
p->add_msg_if_player( _( "You took a selfie." ) );
} else {
if( p->is_blind() ) {
p->add_msg_if_player( _( "You took a photo of %s." ), photo.name );
} else {
p->add_msg_if_player( _( "You took a photo of %1$s. It is %2$s." ), photo.name,
photo_quality_name( photo_quality ) );
}
std::vector<std::string> blinded_names;
for( monster * const &monster_p : monster_vec ) {
if( dist < 4 && one_in( dist + 2 ) && monster_p->has_flag( MF_SEES ) ) {
monster_p->add_effect( effect_blind, rng( 5_turns, 10_turns ) );
blinded_names.push_back( monster_p->name() );
}
}
for( player * const &player_p : player_vec ) {
if( dist < 4 && one_in( dist + 2 ) && !player_p->is_blind() ) {
player_p->add_effect( effect_blind, rng( 5_turns, 10_turns ) );
blinded_names.push_back( player_p->name );
}
}
if( blinded_names.size() > 0 ) {
p->add_msg_if_player( _( "%s looks blinded." ), enumerate_as_string( blinded_names.begin(),
blinded_names.end(), []( const std::string & it ) {
return colorize( it, c_light_blue );
} ) );
}
}
if( monster_vec.size() > 0 ) {
item_save_monsters( *p, *it, monster_vec, photo_quality );
}
return it->type->charges_to_use();
}
}
return it->type->charges_to_use();
}
if( c_photos == choice ) {
show_photo_selection( *p, *it, "CAMERA_EXTENDED_PHOTOS" );
return it->type->charges_to_use();
}
if( c_monsters == choice ) {
if( p->is_blind() ) {
p->add_msg_if_player( _( "You can't see the camera screen, you're blind." ) );
return 0;
}
uilist pmenu;
pmenu.text = _( "Your collection of monsters:" );
std::vector<mtype_id> monster_photos;
std::vector<std::string> descriptions;
std::istringstream f_mon( it->get_var( "CAMERA_MONSTER_PHOTOS" ) );
std::string s;
int k = 0;
while( getline( f_mon, s, ',' ) ) {
if( s.empty() ) {
continue;
}
monster_photos.push_back( mtype_id( s ) );
std::string menu_str;
const monster dummy( monster_photos.back() );
menu_str = dummy.name();
descriptions.push_back( dummy.type->get_description() );
getline( f_mon, s, ',' );
char *chq = &s[0];
const int quality = atoi( chq );
menu_str += " [" + photo_quality_name( quality ) + "]";
pmenu.addentry( k++, true, -1, menu_str.c_str() );
}
int choice;
do {
pmenu.query();
choice = pmenu.ret;
if( choice < 0 ) {
break;
}
popup( "%s", descriptions[choice].c_str() );
} while( true );
return it->type->charges_to_use();
}
if( c_upload == choice ) {
if( p->is_blind() ) {
p->add_msg_if_player( _( "You can't see the camera screen, you're blind." ) );
return 0;
}
p->moves -= to_turns<int>( 2_seconds );
const int inventory_index = g->inv_for_flag( "MC_MOBILE", _( "Insert memory card" ) );
item &mc = p->i_at( inventory_index );
if( mc.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return it->type->charges_to_use();
}
if( !mc.has_flag( "MC_MOBILE" ) ) {
p->add_msg_if_player( m_info, _( "This is not a compatible memory card." ) );
return it->type->charges_to_use();
}
init_memory_card_with_random_stuff( mc );
if( mc.has_flag( "MC_ENCRYPTED" ) ) {
if( !query_yn( _( "This memory card is encrypted. Format and clear data?" ) ) ) {
return it->type->charges_to_use();
}
}
if( mc.has_flag( "MC_HAS_DATA" ) ) {
if( !query_yn( _( "Are you sure you want to clear the old data on the card?" ) ) ) {
return it->type->charges_to_use();
}
}
mc.convert( "mobile_memory_card" );
mc.clear_vars();
mc.unset_flags();
mc.item_tags.insert( "MC_HAS_DATA" );
mc.set_var( "MC_MONSTER_PHOTOS", it->get_var( "CAMERA_MONSTER_PHOTOS" ) );
mc.set_var( "MC_EXTENDED_PHOTOS", it->get_var( "CAMERA_EXTENDED_PHOTOS" ) );
p->add_msg_if_player( m_info,
_( "You upload your photos and monster collection to memory card." ) );
return it->type->charges_to_use();
}
return it->type->charges_to_use();
}
int iuse::ehandcuffs( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) {
if( g->m.has_flag( "SWIMMABLE", pos.x, pos.y ) ) {
it->item_tags.erase( "NO_UNWIELD" );
it->ammo_unset();
it->active = false;
add_msg( m_good, _( "%s automatically turned off!" ), it->tname() );
return it->type->charges_to_use();
}
if( it->charges == 0 ) {
sounds::sound( pos, 2, sounds::sound_t::combat, "Click.", true, "tools", "handcuffs" );
it->item_tags.erase( "NO_UNWIELD" );
it->active = false;
if( p->has_item( *it ) && p->weapon.typeId() == "e_handcuffs" ) {
add_msg( m_good, _( "%s on your hands opened!" ), it->tname() );
}
return it->type->charges_to_use();
}
if( p->has_item( *it ) ) {
if( p->has_active_bionic( bionic_id( "bio_shock" ) ) && p->power_level >= 2 && one_in( 5 ) ) {
p->charge_power( -2 );
it->item_tags.erase( "NO_UNWIELD" );
it->ammo_unset();
it->active = false;
add_msg( m_good, _( "The %s crackle with electricity from your bionic, then come off your hands!" ),
it->tname() );
return it->type->charges_to_use();
}
}
if( calendar::once_every( 1_minutes ) ) {
sounds::sound( pos, 10, sounds::sound_t::alarm, _( "a police siren, whoop WHOOP." ), true,
"environment", "police_siren" );
}
const int x = it->get_var( "HANDCUFFS_X", 0 );
const int y = it->get_var( "HANDCUFFS_Y", 0 );
if( ( it->ammo_remaining() > it->type->maximum_charges() - 1000 ) && ( x != pos.x ||
y != pos.y ) ) {
if( p->has_item( *it ) && p->weapon.typeId() == "e_handcuffs" ) {
if( p->is_elec_immune() ) {
if( one_in( 10 ) ) {
add_msg( m_good, _( "The cuffs try to shock you, but you're protected from electricity." ) );
}
} else {
add_msg( m_bad, _( "Ouch, the cuffs shock you!" ) );
p->apply_damage( nullptr, bp_arm_l, rng( 0, 2 ) );
p->apply_damage( nullptr, bp_arm_r, rng( 0, 2 ) );
p->mod_pain( rng( 2, 5 ) );
}
} else {
add_msg( m_bad, _( "The %s spark with electricity!" ), it->tname() );
}
it->charges -= 50;
if( it->charges < 1 ) {
it->charges = 1;
}
it->set_var( "HANDCUFFS_X", pos.x );
it->set_var( "HANDCUFFS_Y", pos.y );
return it->type->charges_to_use();
}
return it->type->charges_to_use();
}
if( it->active ) {
add_msg( _( "The %s are clamped tightly on your wrists. You can't take them off." ),
it->tname() );
} else {
add_msg( _( "The %s have discharged and can be taken off." ), it->tname() );
}
return it->type->charges_to_use();
}
int iuse::foodperson( player *, item *it, bool t, const tripoint &pos )
{
if( t ) {
if( calendar::once_every( 1_minutes ) ) {
const SpeechBubble &speech = get_speech( "foodperson_mask" );
sounds::sound( pos, speech.volume, sounds::sound_t::alarm, speech.text, true, "speech",
"foodperson_mask" );
}
return it->type->charges_to_use();
}
time_duration shift = time_duration::from_turns( it->magazine_current()->ammo_remaining() *
it->type->tool->turns_per_charge );
add_msg( m_info, _( "Your HUD lights-up: \"Your shift ends in %s\"." ), to_string( shift ) );
return 0;
}
int iuse::radiocar( player *p, item *it, bool, const tripoint & )
{
int choice = -1;
auto bomb_it = std::find_if( it->contents.begin(), it->contents.end(), []( const item & c ) {
return c.has_flag( "RADIOCARITEM" );
} );
if( bomb_it == it->contents.end() ) {
choice = uilist( _( "Using RC car:" ), {
_( "Turn on" ), _( "Put a bomb to car" )
} );
} else {
choice = uilist( _( "Using RC car:" ), {
_( "Turn on" ), bomb_it->tname()
} );
}
if( choice < 0 ) {
return 0;
}
if( choice == 0 ) { //Turn car ON
if( !it->ammo_sufficient() ) {
p->add_msg_if_player( _( "The RC car's batteries seem to be dead." ) );
return 0;
}
it->convert( "radio_car_on" ).active = true;
p->add_msg_if_player(
_( "You turned on your RC car, now place it on ground, and use radio control to play." ) );
return 0;
}
if( choice == 1 ) {
if( bomb_it == it->contents.end() ) { //arming car with bomb
int inventory_index = g->inv_for_flag( "RADIOCARITEM", _( "Arm what?" ) );
item &put = p->i_at( inventory_index );
if( put.is_null() ) {
p->add_msg_if_player( m_info, _( "You do not have that item!" ) );
return 0;
}
if( put.has_flag( "RADIOCARITEM" ) && ( put.volume() <= 1250_ml ||
( put.weight() <= 2_kilogram ) ) ) {
p->moves -= to_turns<int>( 3_seconds );
p->add_msg_if_player( _( "You armed your RC car with %s." ),
put.tname() );
it->put_in( p->i_rem( inventory_index ) );
} else if( !put.has_flag( "RADIOCARITEM" ) ) {
p->add_msg_if_player( _( "RC car with %s ? How?" ),
put.tname() );
} else {
p->add_msg_if_player( _( "Your %s is too heavy or bulky for this RC car." ),
put.tname() );
}
} else { // Disarm the car
p->moves -= to_turns<int>( 2_seconds );
p->inv.assign_empty_invlet( *bomb_it, *p, true ); // force getting an invlet.
p->i_add( *bomb_it );
it->contents.erase( bomb_it );
p->add_msg_if_player( _( "You disarmed your RC car" ) );
}
}
return it->type->charges_to_use();
}
int iuse::radiocaron( player *p, item *it, bool t, const tripoint &pos )
{
if( t ) {
//~Sound of a radio controlled car moving around
sounds::sound( pos, 6, sounds::sound_t::movement, _( "buzzz..." ), true, "misc", "rc_car_drives" );
return it->type->charges_to_use();
} else if( !it->ammo_sufficient() ) {
// Deactivate since other mode has an iuse too.
it->active = false;
return 0;
}
int choice = uilist( _( "What to do with activated RC car?" ), {
_( "Turn off" )
} );
if( choice < 0 ) {
return it->type->charges_to_use();
}
if( choice == 0 ) {
it->convert( "radio_car" ).active = false;
p->add_msg_if_player( _( "You turned off your RC car" ) );
return it->type->charges_to_use();
}
return it->type->charges_to_use();
}
static void sendRadioSignal( player &p, const std::string &signal )
{
for( size_t i = 0; i < p.inv.size(); i++ ) {
item &it = p.inv.find_item( i );
if( it.has_flag( "RADIO_ACTIVATION" ) && it.has_flag( signal ) ) {
sounds::sound( p.pos(), 6, sounds::sound_t::alarm, _( "beep." ), true, "misc", "beep" );
if( it.has_flag( "RADIO_INVOKE_PROC" ) ) {
// Invoke twice: first to transform, then later to proc
it.type->invoke( p, it, p.pos() );
it.ammo_unset();
// The type changed
}
it.type->invoke( p, it, p.pos() );
}
}
g->m.trigger_rc_items( signal );
}
int iuse::radiocontrol( player *p, item *it, bool t, const tripoint & )
{
if( t ) {
if( !it->units_sufficient( *p ) ) {
it->active = false;
p->remove_value( "remote_controlling" );
} else if( p->get_value( "remote_controlling" ).empty() ) {
it->active = false;
}
return it->type->charges_to_use();
}
const char *car_action = nullptr;
if( !it->active ) {
car_action = _( "Take control of RC car" );
} else {
car_action = _( "Stop controlling RC car" );
}
int choice = uilist( _( "What to do with radio control?" ), {
car_action,
_( "Press red button" ), _( "Press blue button" ), _( "Press green button" )
} );
if( choice < 0 ) {
return 0;
} else if( choice == 0 ) {
if( it->active ) {
it->active = false;
p->remove_value( "remote_controlling" );
} else {
std::list<std::pair<tripoint, item *>> rc_pairs = g->m.get_rc_items();
tripoint rc_item_location = {999, 999, 999};
// TODO: grab the closest car or similar?
for( auto &rc_pairs_rc_pair : rc_pairs ) {
if( rc_pairs_rc_pair.second->typeId() == "radio_car_on" &&
rc_pairs_rc_pair.second->active ) {
rc_item_location = rc_pairs_rc_pair.first;
}
}
if( rc_item_location.x == 999 ) {
p->add_msg_if_player( _( "No active RC cars on ground and in range." ) );
return it->type->charges_to_use();
} else {
std::stringstream car_location_string;
// Populate with the point and stash it.
car_location_string << rc_item_location.x << ' ' <<
rc_item_location.y << ' ' << rc_item_location.z;
p->add_msg_if_player( m_good, _( "You take control of the RC car." ) );
p->set_value( "remote_controlling", car_location_string.str() );
it->active = true;
}
}
} else if( choice > 0 ) {
std::string signal = "RADIOSIGNAL_";
std::stringstream choice_str;
choice_str << choice;
signal += choice_str.str();
auto item_list = p->get_radio_items();
for( auto &elem : item_list ) {
if( ( elem )->has_flag( "BOMB" ) && ( elem )->has_flag( signal ) ) {
p->add_msg_if_player( m_warning,
_( "The %s in you inventory would explode on this signal. Place it down before sending the signal." ),
( elem )->display_name() );
return 0;
}
}
p->add_msg_if_player( _( "Click." ) );
sendRadioSignal( *p, signal );
p->moves -= to_turns<int>( 2_seconds );
}
return it->type->charges_to_use();
}
static bool hackveh( player &p, item &it, vehicle &veh )
{
if( !veh.is_locked || !veh.has_security_working() ) {
return true;
}
const bool advanced = !empty( veh.get_avail_parts( "REMOTE_CONTROLS" ) );
if( advanced && veh.is_alarm_on ) {
p.add_msg_if_player( m_bad, _( "This vehicle's security system has locked you out!" ) );
return false;
}
/** @EFFECT_INT increases chance of bypassing vehicle security system */
/** @EFFECT_COMPUTER increases chance of bypassing vehicle security system */
int roll = dice( p.get_skill_level( skill_computer ) + 2, p.int_cur ) - ( advanced ? 50 : 25 );
int effort = 0;
bool success = false;
if( roll < -20 ) { // Really bad rolls will trigger the alarm before you know it exists
effort = 1;
p.add_msg_if_player( m_bad, _( "You trigger the alarm!" ) );
veh.is_alarm_on = true;
} else if( roll >= 20 ) { // Don't bother the player if it's trivial
effort = 1;
p.add_msg_if_player( m_good, _( "You quickly bypass the security system!" ) );
success = true;
}
if( effort == 0 && !query_yn( _( "Try to hack this car's security system?" ) ) ) {
// Scanning for security systems isn't free
p.moves -= to_turns<int>( 1_seconds );
it.charges -= 1;
return false;
}
p.practice( skill_computer, advanced ? 10 : 3 );
if( roll < -10 ) {
effort = rng( 4, 8 );
p.add_msg_if_player( m_bad, _( "You waste some time, but fail to affect the security system." ) );
} else if( roll < 0 ) {
effort = 1;
p.add_msg_if_player( m_bad, _( "You fail to affect the security system." ) );
} else if( roll < 20 ) {
effort = rng( 2, 8 );
p.add_msg_if_player( m_mixed,
_( "You take some time, but manage to bypass the security system!" ) );
success = true;
}
p.moves -= to_turns<int>( time_duration::from_seconds( effort ) );
it.charges -= effort;
if( success && advanced ) { // Unlock controls, but only if they're drive-by-wire
veh.is_locked = false;
}
return success;
}
static vehicle *pickveh( const tripoint &center, bool advanced )
{
static const std::string ctrl = "CTRL_ELECTRONIC";
static const std::string advctrl = "REMOTE_CONTROLS";
uilist pmenu;
pmenu.title = _( "Select vehicle to access" );
std::vector< vehicle * > vehs;
for( auto &veh : g->m.get_vehicles() ) {
auto &v = veh.v;
if( rl_dist( center, v->global_pos3() ) < 40 &&
v->fuel_left( "battery", true ) > 0 &&
( !empty( v->get_avail_parts( advctrl ) ) ||
( !advanced && !empty( v->get_avail_parts( ctrl ) ) ) ) ) {
vehs.push_back( v );
}
}
std::vector<tripoint> locations;
for( int i = 0; i < static_cast<int>( vehs.size() ); i++ ) {
auto veh = vehs[i];
locations.push_back( veh->global_pos3() );
pmenu.addentry( i, true, MENU_AUTOASSIGN, veh->name );
}
if( vehs.empty() ) {
add_msg( m_bad, _( "No vehicle available." ) );
return nullptr;
}
pointmenu_cb callback( locations );
pmenu.callback = &callback;
pmenu.w_y = 0;
pmenu.query();
if( pmenu.ret < 0 || pmenu.ret >= static_cast<int>( vehs.size() ) ) {
return nullptr;
} else {
return vehs[pmenu.ret];
}
}
int iuse::remoteveh( player *p, item *it, bool t, const tripoint &pos )
{
vehicle *remote = g->remoteveh();
if( t ) {
bool stop = false;
if( !it->units_sufficient( *p ) ) {
p->add_msg_if_player( m_bad, _( "The remote control's battery goes dead." ) );
stop = true;
} else if( remote == nullptr ) {
p->add_msg_if_player( _( "Lost contact with the vehicle." ) );
stop = true;
} else if( remote->fuel_left( "battery", true ) == 0 ) {
p->add_msg_if_player( m_bad, _( "The vehicle's battery died." ) );
stop = true;
}
if( stop ) {
it->active = false;
g->setremoteveh( nullptr );
}
return it->type->charges_to_use();
}
bool controlling = it->active && remote != nullptr;
int choice = uilist( _( "What to do with remote vehicle control:" ), {
controlling ? _( "Stop controlling the vehicle." ) : _( "Take control of a vehicle." ),
_( "Execute one vehicle action" )
} );
if( choice < 0 || choice > 1 ) {
return 0;
}
if( choice == 0 && controlling ) {
it->active = false;
g->setremoteveh( nullptr );
return 0;
}
int px = g->u.view_offset.x;
int py = g->u.view_offset.y;
vehicle *veh = pickveh( pos, choice == 0 );
if( veh == nullptr ) {
return 0;
}
if( !hackveh( *p, *it, *veh ) ) {
return 0;
}
if( choice == 0 ) {
it->active = true;
g->setremoteveh( veh );
p->add_msg_if_player( m_good, _( "You take control of the vehicle." ) );
if( !veh->engine_on ) {
veh->start_engines();
}
} else if( choice == 1 ) {
const auto rctrl_parts = veh->get_avail_parts( "REMOTE_CONTROLS" );
// Revert to original behaviour if we can't find remote controls.
if( empty( rctrl_parts ) ) {
veh->use_controls( pos );
} else {
veh->use_controls( rctrl_parts.begin()->pos() );
}
}
g->u.view_offset.x = px;
g->u.view_offset.y = py;
return it->type->charges_to_use();
}
static bool multicooker_hallu( player &p )
{
p.moves -= to_turns<int>( 2_seconds );
const int random_hallu = rng( 1, 7 );
std::vector<tripoint> points;
switch( random_hallu ) {
case 1:
add_msg( m_info, _( "And when you gaze long into a screen, the screen also gazes into you." ) );
return true;
case 2:
add_msg( m_bad, _( "The multi-cooker boiled your head!" ) );
return true;
case 3:
add_msg( m_info, _( "The characters on the screen display an obscene joke. Strange humor." ) );
return true;
case 4:
//~ Single-spaced & lowercase are intentional, conveying hurried speech-KA101
add_msg( m_warning, _( "Are you sure?! the multi-cooker wants to poison your food!" ) );
return true;
case 5:
add_msg( m_info,
_( "The multi-cooker argues with you about the taste preferences. You don't want to deal with it." ) );
return true;
case 6:
for( const tripoint &pt : g->m.points_in_radius( p.pos(), 1 ) ) {
if( g->is_empty( pt ) ) {
points.push_back( pt );
}
}
if( !one_in( 5 ) ) {
add_msg( m_warning, _( "The multi-cooker runs away!" ) );
const tripoint random_point = random_entry( points );
if( monster *const m = g->summon_mon( mon_hallu_multicooker, random_point ) ) {
m->hallucination = true;
m->add_effect( effect_run, 1_turns, num_bp, true );
}
} else {
add_msg( m_bad, _( "You're surrounded by aggressive multi-cookers!" ) );
for( auto &point : points ) {
if( monster *const m = g->summon_mon( mon_hallu_multicooker, point ) ) {
m->hallucination = true;
}
}
}
return true;
default:
return false;
}
}
int iuse::multicooker( player *p, item *it, bool t, const tripoint &pos )
{
static const std::set<std::string> multicooked_subcats = { "CSC_FOOD_MEAT", "CSC_FOOD_VEGGI", "CSC_FOOD_PASTA" };
static const int charges_to_start = 50;
if( t ) {
if( !it->units_sufficient( *p ) ) {
it->active = false;
return 0;
}
int cooktime = it->get_var( "COOKTIME", 0 );
cooktime -= 100;
if( cooktime >= 300 && cooktime < 400 ) {
//Smart or good cook or careful
/** @EFFECT_INT increases chance of checking multi-cooker on time */
/** @EFFECT_SURVIVAL increases chance of checking multi-cooker on time */
if( p->int_cur + p->get_skill_level( skill_cooking ) + p->get_skill_level( skill_survival ) > 16 ) {
add_msg( m_info, _( "The multi-cooker should be finishing shortly..." ) );
}
}
if( cooktime <= 0 ) {
item &meal = it->emplace_back( it->get_var( "DISH" ) );
if( ( *recipe_id( it->get_var( "RECIPE" ) ) ).hot_result() ) {
meal.heat_up();
} else {
meal.set_item_temperature( temp_to_kelvin( std::max( temperatures::cold,
g->weather.get_temperature( pos ) ) ) );
meal.reset_temp_check();
}
it->active = false;
it->erase_var( "DISH" );
it->erase_var( "COOKTIME" );
//~ sound of a multi-cooker finishing its cycle!
sounds::sound( pos, 8, sounds::sound_t::alarm, _( "ding!" ), true, "misc", "ding" );
return 0;
} else {
it->set_var( "COOKTIME", cooktime );
return 0;
}
} else {
enum {
mc_start, mc_stop, mc_take, mc_upgrade
};
if( p->is_underwater() ) {
p->add_msg_if_player( m_info, _( "You can't do that while underwater." ) );
return 0;
}
if( p->has_trait( trait_ILLITERATE ) ) {
add_msg( m_info, _( "You cannot read, and don't understand the screen or the buttons!" ) );
return 0;
}
if( p->has_effect( effect_hallu ) || p->has_effect( effect_visuals ) ) {
if( multicooker_hallu( *p ) ) {
return 0;
}
}
if( p->has_trait( trait_HYPEROPIC ) && !p->worn_with_flag( "FIX_FARSIGHT" ) &&
!p->has_effect( effect_contacts ) ) {
add_msg( m_info, _( "You'll need to put on reading glasses before you can see the screen." ) );
return 0;
}
uilist menu;
menu.text = _( "Welcome to the RobotChef3000. Choose option:" );
// Find actual contents rather than attached mod or battery.
auto dish_it = std::find_if_not( it->contents.begin(), it->contents.end(), []( const item & c ) {
return c.is_toolmod() || c.is_magazine();
} );
if( it->active ) {
menu.addentry( mc_stop, true, 's', _( "Stop cooking" ) );
} else {
if( dish_it == it->contents.end() ) {
if( it->ammo_remaining() < charges_to_start ) {
p->add_msg_if_player( _( "Batteries are low." ) );
return 0;
}
menu.addentry( mc_start, true, 's', _( "Start cooking" ) );
/** @EFFECT_ELECTRONICS >3 allows multicooker upgrade */
/** @EFFECT_FABRICATION >3 allows multicooker upgrade */
if( p->get_skill_level( skill_electronics ) > 3 && p->get_skill_level( skill_fabrication ) > 3 ) {
const auto upgr = it->get_var( "MULTI_COOK_UPGRADE" );
if( upgr.empty() ) {
menu.addentry( mc_upgrade, true, 'u', _( "Upgrade multi-cooker" ) );
} else {
if( upgr == "UPGRADE" ) {
menu.addentry( mc_upgrade, false, 'u', _( "Multi-cooker already upgraded" ) );
} else {
menu.addentry( mc_upgrade, false, 'u', _( "Multi-cooker unable to upgrade" ) );
}
}
}
} else {
menu.addentry( mc_take, true, 't', _( "Take out dish" ) );
}
}
menu.query();
int choice = menu.ret;
if( choice < 0 ) {
return 0;
}
if( mc_stop == choice ) {
if( query_yn( _( "Really stop cooking?" ) ) ) {
it->active = false;
it->erase_var( "DISH" );
it->erase_var( "COOKTIME" );
it->erase_var( "RECIPE" );
}
return 0;
}
if( mc_take == choice ) {
item &dish = *dish_it;
if( dish.has_flag( "FROZEN" ) ) {
dish.cold_up(); //don't know how to check if the dish is frozen liquid and prevent extraction of it into inventory...
}
const std::string dish_name = dish.tname( dish.charges, false );
const bool is_delicious = dish.has_flag( "HOT" ) && dish.has_flag( "EATEN_HOT" );
if( dish.made_of( LIQUID ) ) {
if( !p->check_eligible_containers_for_crafting( *recipe_id( it->get_var( "RECIPE" ) ), 1 ) ) {
p->add_msg_if_player( m_info, _( "You don't have a suitable container to store your %s." ),
dish_name );
return 0;
}
liquid_handler::handle_all_liquid( dish, PICKUP_RANGE );
} else {
p->i_add( dish );
}
it->contents.erase( dish_it );
it->erase_var( "RECIPE" );
if( is_delicious ) {
p->add_msg_if_player( m_good,
_( "You got the dish from the multi-cooker. The %s smells delicious." ),
dish_name );
} else {
p->add_msg_if_player( m_good, _( "You got the %s from the multi-cooker." ),
dish_name );
}
return 0;
}
if( mc_start == choice ) {
uilist dmenu;
dmenu.text = _( "Choose desired meal:" );
std::vector<const recipe *> dishes;
inventory crafting_inv = g->u.crafting_inventory();
//add some tools and qualities. we can't add this qualities to json, because multicook must be used only by activating, not as component other crafts.
crafting_inv.push_back( item( "hotplate", 0 ) ); //hotplate inside
crafting_inv.push_back( item( "tongs", 0 ) ); //some recipes requires tongs
crafting_inv.push_back( item( "toolset", 0 ) ); //toolset with CUT and other qualities inside
crafting_inv.push_back( item( "pot", 0 ) ); //good COOK, BOIL, CONTAIN qualities inside
int counter = 0;
for( const auto &r : g->u.get_learned_recipes().in_category( "CC_FOOD" ) ) {
if( multicooked_subcats.count( r->subcategory ) > 0 ) {
dishes.push_back( r );
const bool can_make = r->requirements().can_make_with_inventory( crafting_inv,
r->get_component_filter() );
dmenu.addentry( counter++, can_make, -1, r->result_name() );
}
}
dmenu.query();
int choice = dmenu.ret;
if( choice < 0 ) {
return 0;
} else {
const recipe *meal = dishes[choice];
int mealtime;
if( it->get_var( "MULTI_COOK_UPGRADE" ) == "UPGRADE" ) {
mealtime = meal->time;
} else {
mealtime = meal->time * 2 ;
}
const int all_charges = charges_to_start + mealtime / ( it->type->tool->turns_per_charge * 100 );
if( it->ammo_remaining() < all_charges ) {
p->add_msg_if_player( m_warning,
_( "The multi-cooker needs %d charges to cook this dish." ),
all_charges );
return 0;
}
auto reqs = meal->requirements();
for( auto it : reqs.get_components() ) {
p->consume_items( it, 1, is_crafting_component );
}
it->set_var( "RECIPE", meal->ident().str() );
it->set_var( "DISH", meal->result() );
it->set_var( "COOKTIME", mealtime );
p->add_msg_if_player( m_good,
_( "The screen flashes blue symbols and scales as the multi-cooker begins to shake." ) );
it->active = true;
it->ammo_consume( charges_to_start, pos );
p->practice( skill_cooking, meal->difficulty * 3 ); //little bonus
return 0;
}
}
if( mc_upgrade == choice ) {
if( !p->has_morale_to_craft() ) {
add_msg( m_info, _( "Your morale is too low to craft..." ) );
return 0;
}
bool has_tools = true;
const inventory &cinv = g->u.crafting_inventory();
if( !cinv.has_amount( "soldering_iron", 1 ) ) {
p->add_msg_if_player( m_warning, _( "You need a %s." ), item::nname( "soldering_iron" ) );
has_tools = false;
}
static const quality_id SCREW_FINE( "SCREW_FINE" );
if( !cinv.has_quality( SCREW_FINE ) ) {
p->add_msg_if_player( m_warning, _( "You need an item with %s of 1 or more to disassemble this." ),
SCREW_FINE.obj().name );
has_tools = false;
}
if( !has_tools ) {
return 0;
}
p->practice( skill_electronics, rng( 5, 10 ) );
p->practice( skill_fabrication, rng( 5, 10 ) );
p->moves -= to_turns<int>( 7_seconds );
/** @EFFECT_INT increases chance to successfully upgrade multi-cooker */
/** @EFFECT_ELECTRONICS increases chance to successfully upgrade multi-cooker */
/** @EFFECT_FABRICATION increases chance to successfully upgrade multi-cooker */
if( p->get_skill_level( skill_electronics ) + p->get_skill_level( skill_fabrication ) + p->int_cur >
rng( 20, 35 ) ) {
p->practice( skill_electronics, rng( 5, 20 ) );
p->practice( skill_fabrication, rng( 5, 20 ) );
p->add_msg_if_player( m_good,
_( "You've successfully upgraded the multi-cooker, master tinkerer! Now it cooks faster!" ) );
it->set_var( "MULTI_COOK_UPGRADE", "UPGRADE" );
return 0;
} else {
if( !one_in( 5 ) ) {
p->add_msg_if_player( m_neutral,
_( "You sagely examine and analyze the multi-cooker, but don't manage to accomplish anything." ) );
} else {
p->add_msg_if_player( m_bad,
_( "Your tinkering nearly breaks the multi-cooker! Fortunately, it still works, but best to stop messing with it." ) );
it->set_var( "MULTI_COOK_UPGRADE", "DAMAGED" );
}
return 0;
}
}
}
return 0;
}
int iuse::cable_attach( player *p, item *it, bool, const tripoint & )
{
std::string initial_state = it->get_var( "state", "attach_first" );
const bool has_bio_cable = p->has_bionic( bionic_id( "bio_cable" ) );
const bool has_solar_pack = p->is_wearing( "solarpack" ) || p->is_wearing( "q_solarpack" );
const bool has_solar_pack_on = p->is_wearing( "solarpack_on" ) || p->is_wearing( "q_solarpack_on" );
const bool wearing_solar_pack = has_solar_pack || has_solar_pack_on;
const auto set_cable_active = []( player * p, item * it, const std::string & state ) {
it->set_var( "state", state );
it->active = true;
it->process( p, p->pos(), false );
p->moves -= 15;
};
if( initial_state == "attach_first" ) {
if( has_bio_cable ) {
uilist kmenu;
kmenu.text = _( "Using cable:" );
kmenu.addentry( 0, true, -1, _( "Attach cable to vehicle" ) );
kmenu.addentry( 1, true, -1, _( "Attach cable to self" ) );
if( wearing_solar_pack ) {
kmenu.addentry( 2, has_solar_pack_on, -1, _( "Attach cable to solar pack" ) );
}
kmenu.query();
int choice = kmenu.ret;
if( choice < 0 ) {
return 0; // we did nothing.
} else if( choice == 1 ) {
set_cable_active( p, it, "cable_charger" );
return 0;
} else if( choice == 2 ) {
set_cable_active( p, it, "solar_pack" );
return 0;
}
// fall through for attaching to a vehicle
}
const cata::optional<tripoint> posp_ = choose_adjacent( _( "Attach cable to vehicle where?" ) );
if( !posp_ ) {
return 0;
}
const tripoint posp = *posp_;
const optional_vpart_position vp = g->m.veh_at( posp );
auto ter = g->m.ter( posp );
if( !vp && ter != t_chainfence ) {
p->add_msg_if_player( _( "There's no vehicle there." ) );
return 0;
} else {
const auto abspos = g->m.getabs( posp );
it->set_var( "source_x", abspos.x );
it->set_var( "source_y", abspos.y );
it->set_var( "source_z", g->get_levz() );
set_cable_active( p, it, "pay_out_cable" );
}
} else {
const auto confirm_source_vehicle = []( player * p, item * it, const bool detach_if_missing ) {
tripoint source_global( it->get_var( "source_x", 0 ),
it->get_var( "source_y", 0 ),
it->get_var( "source_z", 0 ) );
tripoint source_local = g->m.getlocal( source_global );
const optional_vpart_position source_vp = g->m.veh_at( source_local );
vehicle *const source_veh = veh_pointer_or_null( source_vp );
if( detach_if_missing && source_veh == nullptr ) {
if( p != nullptr && p->has_item( *it ) ) {
p->add_msg_if_player( m_bad, _( "You notice the cable has come loose!" ) );
}
it->reset_cable( p );
}
return source_vp;
};
const bool paying_out = initial_state == "pay_out_cable";
const bool cable_cbm = initial_state == "cable_charger";
const bool solar_pack = initial_state == "solar_pack";
bool loose_ends = paying_out || cable_cbm || solar_pack;
uilist kmenu;
kmenu.text = _( "Using cable:" );
kmenu.addentry( 0, paying_out || cable_cbm, -1, _( "Attach loose end of the cable" ) );
kmenu.addentry( 1, true, -1, _( "Detach and re-spool the cable" ) );
if( has_bio_cable && loose_ends ) {
kmenu.addentry( 2, !cable_cbm, -1, _( "Attach cable to self" ) );
// can't attach solar backpacks to cars
if( wearing_solar_pack && cable_cbm ) {
kmenu.addentry( 3, has_solar_pack_on, -1, _( "Attach cable to solar pack" ) );
}
}
kmenu.query();
int choice = kmenu.ret;
if( choice < 0 ) {
return 0; // we did nothing.
} else if( choice == 1 ) {
it->reset_cable( p );
return 0;
} else if( choice == 2 ) {
// connecting self to backpack or car
if( solar_pack ) {
set_cable_active( p, it, "solar_pack_link" );
return 0;
}
const optional_vpart_position source_vp = confirm_source_vehicle( p, it, true );
if( veh_pointer_or_null( source_vp ) != nullptr ) {
set_cable_active( p, it, "cable_charger_link" );
}
return 0;
} else if( choice == 3 ) {
// connecting self to backpack
set_cable_active( p, it, "solar_pack_link" );
return 0;
}
const optional_vpart_position source_vp = confirm_source_vehicle( p, it, paying_out );
vehicle *const source_veh = veh_pointer_or_null( source_vp );
if( source_veh == nullptr && paying_out ) {
return 0;
}
const cata::optional<tripoint> vpos_ = choose_adjacent( _( "Attach cable to vehicle where?" ) );
if( !vpos_ ) {
return 0;
}
const tripoint vpos = *vpos_;
const optional_vpart_position target_vp = g->m.veh_at( vpos );
if( !target_vp ) {
p->add_msg_if_player( _( "There's no vehicle there." ) );
return 0;
} else if( cable_cbm ) {
const auto abspos = g->m.getabs( vpos );
it->set_var( "source_x", abspos.x );
it->set_var( "source_y", abspos.y );
it->set_var( "source_z", g->get_levz() );
set_cable_active( p, it, "cable_charger_link" );
return 0;
} else {
vehicle *const target_veh = &target_vp->vehicle();
if( source_veh == target_veh ) {
if( p != nullptr && p->has_item( *it ) ) {
p->add_msg_if_player( m_warning, _( "The %s already has access to its own electric system!" ),
source_veh->name );
}
return 0;
}
tripoint target_global = g->m.getabs( vpos );
// TODO: make sure there is always a matching vpart id here. Maybe transform this into
// a iuse_actor class, or add a check in item_factory.
const vpart_id vpid( it->typeId() );
point vcoords = source_vp->mount();
vehicle_part source_part( vpid, vcoords, item( *it ) );
source_part.target.first = target_global;
source_part.target.second = g->m.getabs( target_veh->global_pos3() );
source_veh->install_part( vcoords, source_part );
vcoords = target_vp->mount();
vehicle_part target_part( vpid, vcoords, item( *it ) );
tripoint source_global( it->get_var( "source_x", 0 ),
it->get_var( "source_y", 0 ),
it->get_var( "source_z", 0 ) );
target_part.target.first = source_global;
target_part.target.second = g->m.getabs( source_veh->global_pos3() );
target_veh->install_part( vcoords, target_part );
if( p != nullptr && p->has_item( *it ) ) {
p->add_msg_if_player( m_good, _( "You link up the electric systems of the %1$s and the %2$s." ),
source_veh->name, target_veh->name );
}
return 1; // Let the cable be destroyed.
}
}
return 0;
}
int iuse::shavekit( player *p, item *it, bool, const tripoint & )
{
if( !it->ammo_sufficient() ) {
p->add_msg_if_player( _( "You need soap to use this." ) );
} else {
p->assign_activity( activity_id( "ACT_SHAVE" ), 3000 );
}
return it->type->charges_to_use();
}
int iuse::hairkit( player *p, item *it, bool, const tripoint & )
{
p->assign_activity( activity_id( "ACT_HAIRCUT" ), 3000 );
return it->type->charges_to_use();
}
int iuse::weather_tool( player *p, item *it, bool, const tripoint & )
{
const w_point weatherPoint = *g->weather.weather_precise;
/* Possibly used twice. Worth spending the time to precalculate. */
const auto player_local_temp = g->weather.get_temperature( g->u.pos() );
if( it->typeId() == "weather_reader" ) {
p->add_msg_if_player( m_neutral, _( "The %s's monitor slowly outputs the data..." ),
it->tname() );
}
if( it->has_flag( "THERMOMETER" ) ) {
if( it->typeId() == "thermometer" ) {
p->add_msg_if_player( m_neutral, _( "The %1$s reads %2$s." ), it->tname(),
print_temperature( player_local_temp ) );
} else {
p->add_msg_if_player( m_neutral, _( "Temperature: %s." ),
print_temperature( player_local_temp ) );
}
}
if( it->has_flag( "HYGROMETER" ) ) {
if( it->typeId() == "hygrometer" ) {
p->add_msg_if_player(
m_neutral, _( "The %1$s reads %2$s." ), it->tname(),
print_humidity( get_local_humidity( weatherPoint.humidity, g->weather.weather,
g->is_sheltered( g->u.pos() ) ) ) );
} else {
p->add_msg_if_player(
m_neutral, _( "Relative Humidity: %s." ),
print_humidity( get_local_humidity( weatherPoint.humidity, g->weather.weather,
g->is_sheltered( g->u.pos() ) ) ) );
}
}
if( it->has_flag( "BAROMETER" ) ) {
if( it->typeId() == "barometer" ) {
p->add_msg_if_player(
m_neutral, _( "The %1$s reads %2$s." ), it->tname(),
print_pressure( static_cast<int>( weatherPoint.pressure ) ) );
} else {
p->add_msg_if_player( m_neutral, _( "Pressure: %s." ),
print_pressure( static_cast<int>( weatherPoint.pressure ) ) );
}
}
if( it->typeId() == "weather_reader" ) {
int vehwindspeed = 0;
if( optional_vpart_position vp = g->m.veh_at( p->pos() ) ) {
vehwindspeed = abs( vp->vehicle().velocity / 100 ); // For mph
}
const oter_id &cur_om_ter = overmap_buffer.ter( p->global_omt_location() );
/* windpower defined in internal velocity units (=.01 mph) */
const double windpower = 100 * get_local_windpower( g->weather.windspeed + vehwindspeed, cur_om_ter,
p->pos(), g->weather.winddirection, g->is_sheltered( p->pos() ) );
p->add_msg_if_player( m_neutral, _( "Wind Speed: %.1f %s." ),
convert_velocity( windpower, VU_WIND ),
velocity_units( VU_WIND ) );
p->add_msg_if_player(
m_neutral, _( "Feels Like: %s." ),
print_temperature(
get_local_windchill( weatherPoint.temperature, weatherPoint.humidity, windpower / 100 ) +
player_local_temp ) );
std::string dirstring = get_dirstring( g->weather.winddirection );
p->add_msg_if_player( m_neutral, _( "Wind Direction: From the %s." ), dirstring );
}
return 0;
}
int iuse::directional_hologram( player *p, item *it, bool, const tripoint &pos )
{
if( it->is_armor() && !( p->is_worn( *it ) ) ) {
p->add_msg_if_player( m_neutral, _( "You need to wear the %1$s before activating it." ),
it->tname() );
return 0;
}
const cata::optional<tripoint> posp_ = choose_adjacent( _( "Choose hologram direction." ) );
if( !posp_ ) {
return 0;
}
const tripoint posp = *posp_;
if( !g->is_empty( posp ) ) {
p->add_msg_if_player( m_info, _( "Can't create a hologram there." ) );
return 0;
}
monster *const hologram = g->summon_mon( mon_hologram, posp );
tripoint target = pos;
target.x = p->posx() + 2 * SEEX * ( posp.x - p->posx() );
target.y = p->posy() + 2 * SEEY * ( posp.y - p->posy() );
hologram->set_dest( target );
p->mod_moves( -to_turns<int>( 1_seconds ) );
return it->type->charges_to_use();
}
int iuse::capture_monster_veh( player *p, item *it, bool, const tripoint &pos )
{
if( !it->has_flag( "VEHICLE" ) ) {
p->add_msg_if_player( m_info, _( "The %s must be installed in a vehicle before being loaded." ),
it->tname() );
return 0;
}
capture_monster_act( p, it, false, pos );
return 0;
}
int item::release_monster( const tripoint &target, bool spawn )
{
monster new_monster;
try {
::deserialize( new_monster, get_var( "contained_json", "" ) );
} catch( const std::exception &e ) {
debugmsg( _( "Error restoring monster: %s" ), e.what() );
return 0;
}
if( spawn ) {
new_monster.spawn( target );
g->add_zombie( new_monster );
}
erase_var( "contained_name" );
erase_var( "contained_json" );
erase_var( "name" );
erase_var( "weight" );
return 0;
}
// didn't want to drag the monster:: definition into item.h, so just reacquire the monster
// at target
int item::contain_monster( const tripoint &target )
{
const monster *const mon_ptr = g->critter_at<monster>( target );
if( !mon_ptr ) {
return 0;
}
const monster &f = *mon_ptr;
set_var( "contained_json", ::serialize( f ) );
set_var( "contained_name", f.type->nname() );
set_var( "name", string_format( _( "%s holding %s" ), type->nname( 1 ),
f.type->nname() ) );
m_size mon_size = f.get_size();
int new_weight = 0;
switch( mon_size ) {
case MS_TINY:
new_weight = 1000;
break;
case MS_SMALL:
new_weight = 40750;
break;
case MS_MEDIUM:
new_weight = 81500;
break;
case MS_LARGE:
new_weight = 120000;
break;
case MS_HUGE:
new_weight = 200000;
break;
}
set_var( "weight", new_weight );
g->remove_zombie( f );
return 0;
}
int iuse::capture_monster_act( player *p, item *it, bool, const tripoint &pos )
{
if( it->has_var( "contained_name" ) ) {
tripoint target;
if( g->is_empty( pos ) ) {
// It's been activated somewhere where there isn't a player or monster, good.
target = pos;
} else {
if( it->has_flag( "PLACE_RANDOMLY" ) ) {
std::vector<tripoint> valid;
for( const tripoint &dest : g->m.points_in_radius( p->pos(), 1 ) ) {
if( g->is_empty( dest ) ) {
valid.push_back( dest );
}
}
if( valid.empty() ) {
p->add_msg_if_player( _( "There is no place to put the %s." ),
it->get_var( "contained_name", "" ) );
return 0;
}
target = random_entry( valid );
} else {
const std::string query = string_format( _( "Place the %s where?" ),
it->get_var( "contained_name", "" ) );
if( const cata::optional<tripoint> pos_ = choose_adjacent( query ) ) {
target = *pos_;
} else {
return 0;
}
if( !g->is_empty( target ) ) {
p->add_msg_if_player( m_info, _( "You cannot place the %s there!" ),
it->get_var( "contained_name", "" ) );
return 0;
}
}
}
return it->release_monster( target );
} else {
const std::string query = string_format( _( "Capture what with the %s?" ), it->tname() );
const cata::optional<tripoint> target_ = choose_adjacent( query );
if( !target_ ) {
p->add_msg_if_player( m_info, _( "You cannot use a %s there." ), it->tname() );
return 0;
}
const tripoint target = *target_;
// Capture the thing, if it's on the same square.
if( const monster *const mon_ptr = g->critter_at<monster>( target ) ) {
const monster &f = *mon_ptr;
if( !it->has_property( "monster_size_capacity" ) ) {
debugmsg( "%s has no monster_size_capacity.", it->tname() );
return 0;
}
const std::string capacity = it->get_property_string( "monster_size_capacity" );
if( Creature::size_map.count( capacity ) == 0 ) {
debugmsg( "%s has invalid monster_size_capacity %s.",
it->tname(), capacity.c_str() );
return 0;
}
if( f.get_size() > Creature::size_map.find( capacity )->second ) {
p->add_msg_if_player( m_info, _( "The %1$s is too big to put in your %2$s." ),
f.type->nname(), it->tname() );
return 0;
}
// TODO: replace this with some kind of melee check.
int chance = f.hp_percentage() / 10;
// A weaker monster is easier to capture.
// If the monster is friendly, then put it in the item
// without checking if it rolled a success.
if( f.friendly != 0 || one_in( chance ) ) {
return it->contain_monster( target );
} else {
p->add_msg_if_player( m_bad, _( "The %1$s avoids your attempts to put it in the %2$s." ),
f.type->nname(), it->type->nname( 1 ) );
}
p->moves -= to_turns<int>( 1_seconds );
} else {
add_msg( _( "The %s can't capture nothing" ), it->tname() );
return 0;
}
}
return 0;
}
int iuse::ladder( player *p, item *, bool, const tripoint & )
{
if( !g->m.has_zlevels() ) {
debugmsg( "Ladder can't be used in non-z-level mode" );
return 0;
}
const cata::optional<tripoint> pnt_ = choose_adjacent( _( "Put the ladder where?" ) );
if( !pnt_ ) {
return 0;
}
const tripoint pnt = *pnt_;
if( !g->is_empty( pnt ) || g->m.has_furn( pnt ) ) {
p->add_msg_if_player( m_bad, _( "Can't place it there." ) );
return 0;
}
p->add_msg_if_player( _( "You set down the ladder." ) );
p->moves -= to_turns<int>( 5_seconds );
g->m.furn_set( pnt, furn_str_id( "f_ladder" ) );
return 1;
}
washing_requirements washing_requirements_for_volume( const units::volume vol )
{
int water = divide_round_up( vol, 125_ml );
int cleanser = divide_round_up( vol, 1000_ml );
int time = to_turns<int>( 10_seconds * ( vol / 250_ml ) );
return { water, cleanser, time };
}
int iuse::washclothes( player *p, item *, bool, const tripoint & )
{
if( p->fine_detail_vision_mod() > 4 ) {
p->add_msg_if_player( _( "You can't see to do that!" ) );
return 0;
}
// Check that player isn't over volume limit as this might cause it to break... this is a hack.
// TODO: find a better solution.
if( p->volume_capacity() < p->volume_carried() ) {
p->add_msg_if_player( _( "You're carrying too much to clean anything." ) );
return 0;
}
if( p->fine_detail_vision_mod() > 4 ) {
p->add_msg_if_player( _( "You can't see to do that!" ) );
return 0;
}
p->inv.restack( *p );
const inventory &crafting_inv = p->crafting_inventory();
auto is_liquid = []( const item & it ) {
return it.made_of( LIQUID ) || it.contents_made_of( LIQUID );
};
int available_water = std::max(
crafting_inv.charges_of( "water", INT_MAX, is_liquid ),
crafting_inv.charges_of( "clean_water", INT_MAX, is_liquid )
);
int available_cleanser = std::max( crafting_inv.charges_of( "soap" ),
crafting_inv.charges_of( "detergent" ) );
const inventory_filter_preset preset( []( const item_location & location ) {
return location->item_tags.find( "FILTHY" ) != location->item_tags.end();
} );
auto make_raw_stats = [available_water, available_cleanser](
const std::map<const item *, int> &items
) {
units::volume total_volume = 0_ml;
for( const auto &p : items ) {
total_volume += p.first->volume() * p.second;
}
washing_requirements required = washing_requirements_for_volume( total_volume );
auto to_string = []( int val ) -> std::string {
if( val == INT_MAX )
{
return "inf";
}
return string_format( "%3d", val );
};
using stats = inventory_selector::stats;
return stats{{
display_stat( _( "Water" ), required.water, available_water, to_string ),
display_stat( _( "Cleanser" ), required.cleanser, available_cleanser, to_string )
}};
};
// TODO: this should also search surrounding area, not just player inventory.
inventory_iuse_selector inv_s( *p, _( "ITEMS TO CLEAN" ), preset, make_raw_stats );
inv_s.add_character_items( *p );
inv_s.set_title( _( "Multiclean" ) );
inv_s.set_hint( _( "To clean x items, type a number before selecting." ) );
if( inv_s.empty() ) {
popup( std::string( _( "You have nothing to clean." ) ), PF_GET_KEY );
return 0;
}
std::list<std::pair<int, int>> to_clean = inv_s.execute();
if( to_clean.empty() ) {
return 0;
}
// Determine if we have enough water and cleanser for all the items.
units::volume total_volume = 0_ml;
for( std::pair<int, int> pair : to_clean ) {
item i = p->i_at( pair.first );
if( pair.first == INT_MIN ) {
p->add_msg_if_player( m_info, _( "Never mind." ) );
return 0;
}
total_volume += i.volume() * pair.second;
}
washing_requirements required = washing_requirements_for_volume( total_volume );
if( !crafting_inv.has_charges( "water", required.water, is_liquid ) &&
!crafting_inv.has_charges( "water_clean", required.water, is_liquid ) ) {
p->add_msg_if_player( _( "You need %1$i charges of water or clean water to wash these items." ),
required.water );
return 0;
} else if( !crafting_inv.has_charges( "soap", required.cleanser ) &&
!crafting_inv.has_charges( "detergent", required.cleanser ) ) {
p->add_msg_if_player( _( "You need %1$i charges of cleansing agent to wash these items." ),
required.cleanser );
return 0;
}
const std::vector<npc *> helpers = g->u.get_crafting_helpers();
const int helpersize = g->u.get_num_crafting_helpers( 3 );
required.time = required.time * ( 1 - ( helpersize / 10 ) );
for( const npc *np : helpers ) {
add_msg( m_info, _( "%s helps with this task..." ), np->name );
break;
}
// Assign the activity values.
p->assign_activity( activity_id( "ACT_WASH" ), required.time );
for( std::pair<int, int> pair : to_clean ) {
p->activity.values.push_back( pair.first );
p->activity.values.push_back( pair.second );
}
return 0;
}
int iuse::break_stick( player *p, item *it, bool, const tripoint & )
{
p->moves -= to_turns<int>( 2_seconds );
p->mod_stat( "stamina", static_cast<int>( 0.05f * get_option<float>( "PLAYER_MAX_STAMINA" ) ) );
if( p->get_str() < 5 ) {
p->add_msg_if_player( _( "You are too weak to even try." ) );
return 0;
} else if( p->get_str() <= rng( 5, 11 ) ) {
p->add_msg_if_player(
_( "You use all your strength, but the stick won't break. Perhaps try again?" ) );
return 0;
}
std::vector<item_comp> comps;
comps.push_back( item_comp( it->typeId(), 1 ) );
p->consume_items( comps, 1, is_crafting_component );
int chance = rng( 0, 100 );
if( chance <= 20 ) {
p->add_msg_if_player( _( "You try to break the stick in two, but it shatters into splinters." ) );
g->m.spawn_item( p->pos(), "splinter", 2 );
return 1;
} else if( chance <= 40 ) {
p->add_msg_if_player( _( "The stick breaks clean into two parts." ) );
g->m.spawn_item( p->pos(), "stick", 2 );
return 1;
} else if( chance <= 100 ) {
p->add_msg_if_player( _( "You break the stick, but one half shatters into splinters." ) );
g->m.spawn_item( p->pos(), "stick", 1 );
g->m.spawn_item( p->pos(), "splinter", 1 );
return 1;
}
return 0;
}
int iuse::weak_antibiotic( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You take some %s." ), it->tname() );
if( p->has_effect( effect_infected ) && !p->has_effect( effect_weak_antibiotic ) ) {
p->add_msg_if_player( m_good, _( "The throbbing of the infection diminishes. Slightly." ) );
}
p->add_effect( effect_weak_antibiotic, 12_hours );
p->add_effect( effect_weak_antibiotic_visible, rng( 9_hours, 15_hours ) );
return it->type->charges_to_use();
}
int iuse::strong_antibiotic( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You take some %s." ), it->tname() );
if( p->has_effect( effect_infected ) && !p->has_effect( effect_strong_antibiotic ) ) {
p->add_msg_if_player( m_good, _( "You feel much better - almost entirely." ) );
}
p->add_effect( effect_strong_antibiotic, 12_hours );
p->add_effect( effect_strong_antibiotic_visible, rng( 9_hours, 15_hours ) );
return it->type->charges_to_use();
}
int iuse::panacea( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You take some %s." ), it->tname() );
if( !p->has_effect( effect_panacea ) ) {
p->add_msg_if_player( m_good, _( "You feel AMAZING!" ) );
}
p->add_effect( effect_panacea, 1_minutes );
return it->type->charges_to_use();
}
int iuse::craft( player *p, item *it, bool, const tripoint & )
{
if( !p->is_wielding( *it ) ) {
if( !p->is_armed() || query_yn( "Wield the %s and start working?", it->tname() ) ) {
if( !p->wield( *it ) ) {
// Will likely happen if the in progress craft is too heavy, or the player is
// wielding something that can't be unwielded
return 0;
}
} else {
return 0;
}
}
const std::string craft_name = p->weapon.tname();
if( !p->weapon.is_craft() ) {
debugmsg( "Attempted to start working on non craft '%s.' Aborting.", craft_name );
return 0;
}
if( !p->can_continue_craft( p->weapon ) ) {
return 0;
}
const recipe &rec = it->get_making();
if( p->has_recipe( &rec, p->crafting_inventory(), p->get_crafting_helpers() ) == -1 ) {
p->add_msg_player_or_npc(
string_format( _( "You don't know the recipe for the %s and can't continue crafting." ),
rec.result_name() ),
string_format( _( "<npcname> doesn't know the recipe for the %s and can't continue crafting." ),
rec.result_name() )
);
return 0;
}
p->add_msg_player_or_npc(
string_format( pgettext( "in progress craft", "You start working on the %s." ), craft_name ),
string_format( pgettext( "in progress craft", "<npcname> starts working on the %s." ),
craft_name ) );
p->assign_activity( activity_id( "ACT_CRAFT" ) );
p->activity.targets.push_back( item_location( *p, &p->weapon ) );
p->activity.values.push_back( 0 ); // Not a long craft
return 0;
}
int iuse::disassemble( player *p, item *it, bool, const tripoint & )
{
if( !p->has_item( *it ) ) {
return 0;
}
p->disassemble( item_location( *p, it ), false );
return 0;
}
static int gobag( player *p, item *it, const bool is_personal )
{
std::vector<item> items = item_group::items_from( "gobag_contents", calendar::turn );
item last_armor;
p->add_msg_if_player( _( "You empty the contents of the go bag onto the floor." ) );
for( item &content : items ) {
if( content.is_armor() ) {
if( is_personal && !content.has_flag( "FIT" ) ) {
content.set_flag( "FIT" );
} else if( content.typeId() == last_armor.typeId() ) {
if( last_armor.has_flag( "FIT" ) && !content.has_flag( "FIT" ) ) {
content.set_flag( "FIT" );
} else if( !last_armor.has_flag( "FIT" ) && content.has_flag( "FIT" ) ) {
content.unset_flag( "FIT" );
}
}
last_armor = content;
}
if( units::to_liter( content.get_storage() ) >= 10.0 && it->has_flag( "FILTHY" ) ) {
content.set_flag( "FILTHY" );
}
g->m.add_item_or_charges( p->pos(), content );
}
p->i_rem( it );
return 0;
}
int iuse::gobag_normal( player *p, item *it, bool, const tripoint & )
{
return gobag( p, it, false );
}
int iuse::gobag_personal( player *p, item *it, bool, const tripoint & )
{
return gobag( p, it, true );
}
int iuse::magnesium_tablet( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( _( "You pop a %s." ), it->tname() );
if( p->has_effect( effect_magnesium_supplements ) ) {
p->add_msg_if_player( m_warning,
_( "Simply taking more magnesium won't help. You have to go to sleep for it to work." ) );
}
p->add_effect( effect_magnesium_supplements, 16_hours );
return it->type->charges_to_use();
}
int iuse::coin_flip( player *p, item *it, bool, const tripoint & )
{
p->add_msg_if_player( m_info, _( "You flip a %s." ), it->tname() );
p->add_msg_if_player( m_info, one_in( 2 ) ? _( "Heads!" ) : _( "Tails!" ) );
return 0;
}
int iuse::magic_8_ball( player *p, item *it, bool, const tripoint & )
{
enum {
BALL8_GOOD,
BALL8_UNK = 10,
BALL8_BAD = 15
};
static const std::array<const char *, 20> tab = {{
translate_marker( "It is certain." ),
translate_marker( "It is decidedly so." ),
translate_marker( "Without a doubt." ),
translate_marker( "Yes - definitely." ),
translate_marker( "You may rely on it." ),
translate_marker( "As I see it, yes." ),
translate_marker( "Most likely." ),
translate_marker( "Outlook good." ),
translate_marker( "Yes." ),
translate_marker( "Signs point to yes." ),
translate_marker( "Reply hazy, try again." ),
translate_marker( "Ask again later." ),
translate_marker( "Better not tell you now." ),
translate_marker( "Cannot predict now." ),
translate_marker( "Concentrate and ask again." ),
translate_marker( "Don't count on it." ),
translate_marker( "My reply is no." ),
translate_marker( "My sources say no." ),
translate_marker( "Outlook not so good." ),
translate_marker( "Very doubtful." )
}
};
p->add_msg_if_player( m_info, _( "You ask the %s, then flip it." ), it->tname() );
int rn = rng( 0, tab.size() - 1 );
auto color = ( rn >= BALL8_BAD ? m_bad : rn >= BALL8_UNK ? m_info : m_good );
p->add_msg_if_player( color, _( "The %s says: %s" ), it->tname(), _( tab[rn] ) );
return 0;
}
use_function::use_function( const use_function &other )
: actor( other.actor ? other.actor->clone() : nullptr )
{
}
use_function &use_function::operator=( iuse_actor *const f )
{
return operator=( use_function( f ) );
}
use_function &use_function::operator=( const use_function &other )
{
actor.reset( other.actor ? other.actor->clone() : nullptr );
return *this;
}
void use_function::dump_info( const item &it, std::vector<iteminfo> &dump ) const
{
if( actor != nullptr ) {
actor->info( it, dump );
}
}
ret_val<bool> use_function::can_call( const player &p, const item &it, bool t,
const tripoint &pos ) const
{
if( actor == nullptr ) {
return ret_val<bool>::make_failure( _( "You can't do anything interesting with your %s." ),
it.tname() );
}
return actor->can_use( p, it, t, pos );
}
int use_function::call( player &p, item &it, bool active, const tripoint &pos ) const
{
return actor->use( p, it, active, pos );
}
You can’t perform that action at this time.