#include "debug_menu.h" #include // IWYU pragma: no_include // IWYU pragma: no_include #include #include #include #include #include #include // IWYU pragma: keep #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "achievement.h" #include "action.h" #include "avatar.h" #include "bodypart.h" #include "calendar.h" #include "cata_utility.h" #include "catacharset.h" #include "character.h" #include "character_id.h" #include "character_martial_arts.h" #include "clzones.h" #include "color.h" #include "coordinates.h" #include "creature.h" #include "debug.h" #include "dialogue_chatbin.h" #include "effect.h" #include "effect_source.h" #include "enum_conversions.h" #include "enums.h" #include "event.h" #include "event_bus.h" #include "faction.h" #include "filesystem.h" // IWYU pragma: keep #include "game.h" #include "game_constants.h" #include "game_inventory.h" #include "input.h" #include "inventory.h" #include "item.h" #include "item_group.h" #include "item_location.h" #include "itype.h" #include "location.h" #include "magic.h" #include "map.h" #include "map_extras.h" #include "mapgen.h" #include "mapgendata.h" #include "martialarts.h" #include "memory_fast.h" #include "messages.h" #include "mission.h" #include "monster.h" #include "monstergenerator.h" #include "morale_types.h" #include "mtype.h" #include "npc.h" #include "npc_class.h" #include "omdata.h" #include "optional.h" #include "options.h" #include "output.h" #include "overmap.h" #include "overmap_ui.h" #include "overmapbuffer.h" #include "path_info.h" // IWYU pragma: keep #include "pimpl.h" #include "player.h" #include "point.h" #include "popup.h" #include "recipe_dictionary.h" #include "rng.h" #include "sounds.h" #include "stomach.h" #include "string_formatter.h" #include "string_input_popup.h" #include "trait_group.h" #include "translations.h" #include "type_id.h" #include "ui.h" #include "ui_manager.h" #include "units.h" #include "veh_type.h" #include "vehicle.h" #include "vitamin.h" #include "vpart_position.h" #include "weather.h" #include "weather_gen.h" #include "weather_type.h" #include "weighted_list.h" static const efftype_id effect_asthma( "asthma" ); static const efftype_id effect_flu( "flu" ); static const mtype_id mon_generator( "mon_generator" ); static const trait_id trait_ASTHMA( "ASTHMA" ); #if defined(TILES) #include "sdl_wrappers.h" #endif #define dbg(x) DebugLog((x),D_GAME) << __FILE__ << ":" << __LINE__ << ": " namespace io { template<> std::string enum_to_string( debug_menu::debug_menu_index v ) { switch( v ) { // *INDENT-OFF* case debug_menu::debug_menu_index::WISH: return "WISH"; case debug_menu::debug_menu_index::SHORT_TELEPORT: return "SHORT_TELEPORT"; case debug_menu::debug_menu_index::LONG_TELEPORT: return "LONG_TELEPORT"; case debug_menu::debug_menu_index::REVEAL_MAP: return "REVEAL_MAP"; case debug_menu::debug_menu_index::SPAWN_NPC: return "SPAWN_NPC"; case debug_menu::debug_menu_index::SPAWN_MON: return "SPAWN_MON"; case debug_menu::debug_menu_index::GAME_STATE: return "GAME_STATE"; case debug_menu::debug_menu_index::KILL_NPCS: return "KILL_NPCS"; case debug_menu::debug_menu_index::MUTATE: return "MUTATE"; case debug_menu::debug_menu_index::SPAWN_VEHICLE: return "SPAWN_VEHICLE"; case debug_menu::debug_menu_index::CHANGE_SKILLS: return "CHANGE_SKILLS"; case debug_menu::debug_menu_index::LEARN_MA: return "LEARN_MA"; case debug_menu::debug_menu_index::UNLOCK_RECIPES: return "UNLOCK_RECIPES"; case debug_menu::debug_menu_index::EDIT_PLAYER: return "EDIT_PLAYER"; case debug_menu::debug_menu_index::SPAWN_ARTIFACT: return "SPAWN_ARTIFACT"; case debug_menu::debug_menu_index::SPAWN_CLAIRVOYANCE: return "SPAWN_CLAIRVOYANCE"; case debug_menu::debug_menu_index::MAP_EDITOR: return "MAP_EDITOR"; case debug_menu::debug_menu_index::CHANGE_WEATHER: return "CHANGE_WEATHER"; case debug_menu::debug_menu_index::WIND_DIRECTION: return "WIND_DIRECTION"; case debug_menu::debug_menu_index::WIND_SPEED: return "WIND_SPEED"; case debug_menu::debug_menu_index::GEN_SOUND: return "GEN_SOUND"; case debug_menu::debug_menu_index::KILL_MONS: return "KILL_MONS"; case debug_menu::debug_menu_index::DISPLAY_HORDES: return "DISPLAY_HORDES"; case debug_menu::debug_menu_index::TEST_IT_GROUP: return "TEST_IT_GROUP"; case debug_menu::debug_menu_index::DAMAGE_SELF: return "DAMAGE_SELF"; case debug_menu::debug_menu_index::BLEED_SELF: return "BLEED_SELF"; case debug_menu::debug_menu_index::SHOW_SOUND: return "SHOW_SOUND"; case debug_menu::debug_menu_index::DISPLAY_WEATHER: return "DISPLAY_WEATHER"; case debug_menu::debug_menu_index::DISPLAY_SCENTS: return "DISPLAY_SCENTS"; case debug_menu::debug_menu_index::CHANGE_TIME: return "CHANGE_TIME"; case debug_menu::debug_menu_index::SET_AUTOMOVE: return "SET_AUTOMOVE"; case debug_menu::debug_menu_index::SHOW_MUT_CAT: return "SHOW_MUT_CAT"; case debug_menu::debug_menu_index::OM_EDITOR: return "OM_EDITOR"; case debug_menu::debug_menu_index::BENCHMARK: return "BENCHMARK"; case debug_menu::debug_menu_index::OM_TELEPORT: return "OM_TELEPORT"; case debug_menu::debug_menu_index::OM_TELEPORT_COORDINATES: return "OM_TELEPORT_COORDINATES"; case debug_menu::debug_menu_index::TRAIT_GROUP: return "TRAIT_GROUP"; case debug_menu::debug_menu_index::ENABLE_ACHIEVEMENTS: return "ENABLE_ACHIEVEMENTS"; case debug_menu::debug_menu_index::SHOW_MSG: return "SHOW_MSG"; case debug_menu::debug_menu_index::CRASH_GAME: return "CRASH_GAME"; case debug_menu::debug_menu_index::MAP_EXTRA: return "MAP_EXTRA"; case debug_menu::debug_menu_index::DISPLAY_NPC_PATH: return "DISPLAY_NPC_PATH"; case debug_menu::debug_menu_index::PRINT_FACTION_INFO: return "PRINT_FACTION_INFO"; case debug_menu::debug_menu_index::PRINT_NPC_MAGIC: return "PRINT_NPC_MAGIC"; case debug_menu::debug_menu_index::QUIT_NOSAVE: return "QUIT_NOSAVE"; case debug_menu::debug_menu_index::TEST_WEATHER: return "TEST_WEATHER"; case debug_menu::debug_menu_index::SAVE_SCREENSHOT: return "SAVE_SCREENSHOT"; case debug_menu::debug_menu_index::GAME_REPORT: return "GAME_REPORT"; case debug_menu::debug_menu_index::DISPLAY_SCENTS_LOCAL: return "DISPLAY_SCENTS_LOCAL"; case debug_menu::debug_menu_index::DISPLAY_SCENTS_TYPE_LOCAL: return "DISPLAY_SCENTS_TYPE_LOCAL"; case debug_menu::debug_menu_index::DISPLAY_TEMP: return "DISPLAY_TEMP"; case debug_menu::debug_menu_index::DISPLAY_VEHICLE_AI: return "DISPLAY_VEHICLE_AI"; case debug_menu::debug_menu_index::DISPLAY_VISIBILITY: return "DISPLAY_VISIBILITY"; case debug_menu::debug_menu_index::DISPLAY_LIGHTING: return "DISPLAY_LIGHTING"; case debug_menu::debug_menu_index::DISPLAY_TRANSPARENCY: return "DISPLAY_TRANSPARENCY"; case debug_menu::debug_menu_index::DISPLAY_REACHABILITY_ZONES: return "DISPLAY_REACHABILITY_ZONES"; case debug_menu::debug_menu_index::DISPLAY_RADIATION: return "DISPLAY_RADIATION"; case debug_menu::debug_menu_index::HOUR_TIMER: return "HOUR_TIMER"; case debug_menu::debug_menu_index::LEARN_SPELLS: return "LEARN_SPELLS"; case debug_menu::debug_menu_index::LEVEL_SPELLS: return "LEVEL_SPELLS"; case debug_menu::debug_menu_index::TEST_MAP_EXTRA_DISTRIBUTION: return "TEST_MAP_EXTRA_DISTRIBUTION"; case debug_menu::debug_menu_index::NESTED_MAPGEN: return "NESTED_MAPGEN"; case debug_menu::debug_menu_index::VEHICLE_BATTERY_CHARGE: return "VEHICLE_BATTERY_CHARGE"; case debug_menu::debug_menu_index::GENERATE_EFFECT_LIST: return "GENERATE_EFFECT_LIST"; // *INDENT-ON* case debug_menu::debug_menu_index::last: break; } debugmsg( "unknown debug_menu::debug_menu_index %d", static_cast( v ) ); return ""; } } // namespace io namespace debug_menu { class mission_debug { private: // Doesn't actually "destroy" the mission, just removes assignments static void remove_mission( mission &m ); public: static void edit_mission( mission &m ); static void edit( player &who ); static void edit_player(); static void edit_npc( npc &who ); static std::string describe( const mission &m ); }; static int player_uilist() { std::vector uilist_initializer = { { uilist_entry( debug_menu_index::MUTATE, true, 'M', _( "Mutate" ) ) }, { uilist_entry( debug_menu_index::CHANGE_SKILLS, true, 's', _( "Change all skills" ) ) }, { uilist_entry( debug_menu_index::LEARN_MA, true, 'l', _( "Learn all melee styles" ) ) }, { uilist_entry( debug_menu_index::UNLOCK_RECIPES, true, 'r', _( "Unlock all recipes" ) ) }, { uilist_entry( debug_menu_index::EDIT_PLAYER, true, 'p', _( "Edit player/NPC" ) ) }, { uilist_entry( debug_menu_index::DAMAGE_SELF, true, 'd', _( "Damage self" ) ) }, { uilist_entry( debug_menu_index::BLEED_SELF, true, 'b', _( "Bleed self" ) ) }, { uilist_entry( debug_menu_index::SET_AUTOMOVE, true, 'a', _( "Set automove route" ) ) }, }; if( !spell_type::get_all().empty() ) { uilist_initializer.emplace_back( uilist_entry( debug_menu_index::LEARN_SPELLS, true, 'S', _( "Learn all spells" ) ) ); uilist_initializer.emplace_back( uilist_entry( debug_menu_index::LEVEL_SPELLS, true, 'L', _( "Level a spell" ) ) ); } return uilist( _( "Player…" ), uilist_initializer ); } static int info_uilist( bool display_all_entries = true ) { // always displayed std::vector uilist_initializer = { { uilist_entry( debug_menu_index::SAVE_SCREENSHOT, true, 'H', _( "Take screenshot" ) ) }, { uilist_entry( debug_menu_index::GAME_REPORT, true, 'r', _( "Generate game report" ) ) }, }; if( display_all_entries ) { const std::vector debug_only_options = { { uilist_entry( debug_menu_index::GAME_STATE, true, 'g', _( "Check game state" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_HORDES, true, 'h', _( "Display hordes" ) ) }, { uilist_entry( debug_menu_index::TEST_IT_GROUP, true, 'i', _( "Test item group" ) ) }, { uilist_entry( debug_menu_index::SHOW_SOUND, true, 'c', _( "Show sound clustering" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_WEATHER, true, 'w', _( "Display weather" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_SCENTS, true, 'S', _( "Display overmap scents" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_SCENTS_LOCAL, true, 's', _( "Toggle display local scents" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_SCENTS_TYPE_LOCAL, true, 'y', _( "Toggle display local scents type" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_TEMP, true, 'T', _( "Toggle display temperature" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_VEHICLE_AI, true, 'V', _( "Toggle display vehicle autopilot overlay" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_VISIBILITY, true, 'v', _( "Toggle display visibility" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_LIGHTING, true, 'l', _( "Toggle display lighting" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_TRANSPARENCY, true, 'p', _( "Toggle display transparency" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_REACHABILITY_ZONES, true, 'z', _( "Toggle display reachability zones" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_RADIATION, true, 'R', _( "Toggle display radiation" ) ) }, { uilist_entry( debug_menu_index::SHOW_MUT_CAT, true, 'm', _( "Show mutation category levels" ) ) }, { uilist_entry( debug_menu_index::BENCHMARK, true, 'b', _( "Draw benchmark (X seconds)" ) ) }, { uilist_entry( debug_menu_index::HOUR_TIMER, true, 'E', _( "Toggle hour timer" ) ) }, { uilist_entry( debug_menu_index::TRAIT_GROUP, true, 't', _( "Test trait group" ) ) }, { uilist_entry( debug_menu_index::DISPLAY_NPC_PATH, true, 'n', _( "Toggle NPC pathfinding on map" ) ) }, { uilist_entry( debug_menu_index::PRINT_FACTION_INFO, true, 'f', _( "Print faction info to console" ) ) }, { uilist_entry( debug_menu_index::PRINT_NPC_MAGIC, true, 'M', _( "Print NPC magic info to console" ) ) }, { uilist_entry( debug_menu_index::TEST_WEATHER, true, 'W', _( "Test weather" ) ) }, { uilist_entry( debug_menu_index::TEST_MAP_EXTRA_DISTRIBUTION, true, 'e', _( "Test map extra list" ) ) }, { uilist_entry( debug_menu_index::GENERATE_EFFECT_LIST, true, 'L', _( "Generate effect list" ) ) }, }; uilist_initializer.insert( uilist_initializer.begin(), debug_only_options.begin(), debug_only_options.end() ); } return uilist( _( "Info…" ), uilist_initializer ); } static int game_uilist() { std::vector uilist_initializer = { { uilist_entry( debug_menu_index::ENABLE_ACHIEVEMENTS, true, 'a', _( "Enable achievements" ) ) }, { uilist_entry( debug_menu_index::SHOW_MSG, true, 'd', _( "Show debug message" ) ) }, { uilist_entry( debug_menu_index::CRASH_GAME, true, 'C', _( "Crash game (test crash handling)" ) ) }, { uilist_entry( debug_menu_index::QUIT_NOSAVE, true, 'Q', _( "Quit to main menu" ) ) }, }; return uilist( _( "Game…" ), uilist_initializer ); } static int vehicle_uilist() { std::vector uilist_initializer = { { uilist_entry( debug_menu_index::VEHICLE_BATTERY_CHARGE, true, 'b', _( "Change [b]attery charge" ) ) }, }; return uilist( _( "Vehicle…" ), uilist_initializer ); } static int teleport_uilist() { const std::vector uilist_initializer = { { uilist_entry( debug_menu_index::SHORT_TELEPORT, true, 's', _( "Teleport - short range" ) ) }, { uilist_entry( debug_menu_index::LONG_TELEPORT, true, 'l', _( "Teleport - long range" ) ) }, { uilist_entry( debug_menu_index::OM_TELEPORT, true, 'o', _( "Teleport - adjacent overmap" ) ) }, { uilist_entry( debug_menu_index::OM_TELEPORT_COORDINATES, true, 'p', _( "Teleport - specific overmap coordinates" ) ) }, }; return uilist( _( "Teleport…" ), uilist_initializer ); } static int spawning_uilist() { const std::vector uilist_initializer = { { uilist_entry( debug_menu_index::WISH, true, 'w', _( "Spawn an item" ) ) }, { uilist_entry( debug_menu_index::SPAWN_NPC, true, 'n', _( "Spawn NPC" ) ) }, { uilist_entry( debug_menu_index::SPAWN_MON, true, 'm', _( "Spawn monster" ) ) }, { uilist_entry( debug_menu_index::SPAWN_VEHICLE, true, 'v', _( "Spawn a vehicle" ) ) }, { uilist_entry( debug_menu_index::SPAWN_ARTIFACT, true, 'a', _( "Spawn artifact" ) ) }, { uilist_entry( debug_menu_index::SPAWN_CLAIRVOYANCE, true, 'c', _( "Spawn clairvoyance artifact" ) ) }, }; return uilist( _( "Spawning…" ), uilist_initializer ); } static int map_uilist() { const std::vector uilist_initializer = { { uilist_entry( debug_menu_index::REVEAL_MAP, true, 'r', _( "Reveal map" ) ) }, { uilist_entry( debug_menu_index::KILL_NPCS, true, 'k', _( "Kill NPCs" ) ) }, { uilist_entry( debug_menu_index::MAP_EDITOR, true, 'M', _( "Map editor" ) ) }, { uilist_entry( debug_menu_index::CHANGE_WEATHER, true, 'w', _( "Change weather" ) ) }, { uilist_entry( debug_menu_index::WIND_DIRECTION, true, 'd', _( "Change wind direction" ) ) }, { uilist_entry( debug_menu_index::WIND_SPEED, true, 's', _( "Change wind speed" ) ) }, { uilist_entry( debug_menu_index::GEN_SOUND, true, 'S', _( "Generate sound" ) ) }, { uilist_entry( debug_menu_index::KILL_MONS, true, 'K', _( "Kill all monsters" ) ) }, { uilist_entry( debug_menu_index::CHANGE_TIME, true, 't', _( "Change time" ) ) }, { uilist_entry( debug_menu_index::OM_EDITOR, true, 'O', _( "Overmap editor" ) ) }, { uilist_entry( debug_menu_index::MAP_EXTRA, true, 'm', _( "Spawn map extra" ) ) }, { uilist_entry( debug_menu_index::NESTED_MAPGEN, true, 'n', _( "Spawn nested mapgen" ) ) }, }; return uilist( _( "Map…" ), uilist_initializer ); } /** * Create the debug menu UI list. * @param display_all_entries: `true` if all entries should be displayed, `false` is some entries should be hidden (for ex. when the debug menu is called from the main menu). * This allows to have some menu elements at the same time in the main menu and in the debug menu. * @returns The chosen action. */ static cata::optional debug_menu_uilist( bool display_all_entries = true ) { std::vector menu = { { uilist_entry( 1, true, 'i', _( "Info…" ) ) }, }; if( display_all_entries ) { const std::vector debug_menu = { { uilist_entry( 6, true, 'g', _( "Game…" ) ) }, { uilist_entry( 2, true, 's', _( "Spawning…" ) ) }, { uilist_entry( 3, true, 'p', _( "Player…" ) ) }, { uilist_entry( 7, true, 'v', _( "Vehicle…" ) ) }, { uilist_entry( 4, true, 't', _( "Teleport…" ) ) }, { uilist_entry( 5, true, 'm', _( "Map…" ) ) }, }; // insert debug-only menu right after "Info". menu.insert( menu.begin() + 1, debug_menu.begin(), debug_menu.end() ); } std::string msg; if( display_all_entries ) { msg = _( "Debug Functions - Using these will cheat not only the game, but yourself.\nYou won't grow. You won't improve.\nTaking this shortcut will gain you nothing. Your victory will be hollow.\nNothing will be risked and nothing will be gained." ); } else { msg = _( "Debug Functions" ); } while( true ) { const int group = uilist( msg, menu ); int action; switch( group ) { case 1: action = info_uilist( display_all_entries ); break; case 2: action = spawning_uilist(); break; case 3: action = player_uilist(); break; case 4: action = teleport_uilist(); break; case 5: action = map_uilist(); break; case 6: action = game_uilist(); break; case 7: action = vehicle_uilist(); break; default: return cata::nullopt; } if( action >= 0 ) { return static_cast( action ); } else { return cata::nullopt; } } } void teleport_short() { const cata::optional where = g->look_around(); location &player_location = get_player_location(); if( !where || *where == player_location.pos() ) { return; } g->place_player( *where ); const tripoint new_pos( player_location.pos() ); add_msg( _( "You teleport to point (%d,%d,%d)." ), new_pos.x, new_pos.y, new_pos.z ); } void teleport_long() { const tripoint_abs_omt where( ui::omap::choose_point() ); if( where == overmap::invalid_tripoint ) { return; } g->place_player_overmap( where ); add_msg( _( "You teleport to submap (%s)." ), where.to_string() ); } void teleport_overmap( bool specific_coordinates ) { Character &player_character = get_player_character(); tripoint_abs_omt where; if( specific_coordinates ) { const std::string text = string_input_popup() .title( "Teleport where?" ) .width( 20 ) .query_string(); if( text.empty() ) { return; } const std::vector coord_strings = string_split( text, ',' ); tripoint coord; coord.x = !coord_strings.empty() ? std::atoi( coord_strings[0].c_str() ) : 0; coord.y = coord_strings.size() >= 2 ? std::atoi( coord_strings[1].c_str() ) : 0; coord.z = coord_strings.size() >= 3 ? std::atoi( coord_strings[2].c_str() ) : 0; where = tripoint_abs_omt( OMAPX * coord.x, OMAPY * coord.y, coord.z ); } else { const cata::optional dir_ = choose_direction( _( "Where is the desired overmap?" ) ); if( !dir_ ) { return; } const tripoint offset = tripoint( OMAPX * dir_->x, OMAPY * dir_->y, dir_->z ); where = player_character.global_omt_location() + offset; } g->place_player_overmap( where ); const tripoint_abs_om new_pos = project_to( player_character.global_omt_location() ); add_msg( _( "You teleport to overmap %s." ), new_pos.to_string() ); } void spawn_nested_mapgen() { uilist nest_menu; std::vector nest_str; for( auto &nested : nested_mapgen ) { nest_menu.addentry( -1, true, -1, nested.first ); nest_str.push_back( nested.first ); } nest_menu.query(); const int nest_choice = nest_menu.ret; if( nest_choice >= 0 && nest_choice < static_cast( nest_str.size() ) ) { const cata::optional where = g->look_around(); if( !where ) { return; } map &here = get_map(); const tripoint_abs_ms abs_ms( here.getabs( *where ) ); const tripoint_abs_omt abs_omt = project_to( abs_ms ); const tripoint_abs_sm abs_sub = project_to( abs_ms ); map target_map; target_map.load( abs_sub, true ); // TODO: fix point types const tripoint local_ms = target_map.getlocal( abs_ms.raw() ); mapgendata md( abs_omt, target_map, 0.0f, calendar::turn, nullptr ); const auto &ptr = nested_mapgen[nest_str[nest_choice]].pick(); if( ptr == nullptr ) { return; } ( *ptr )->nest( md, local_ms.xy() ); target_map.save(); g->load_npcs(); here.invalidate_map_cache( here.get_abs_sub().z ); } } void character_edit_menu() { std::vector< tripoint > locations; uilist charmenu; int charnum = 0; avatar &player_character = get_avatar(); charmenu.addentry( charnum++, true, MENU_AUTOASSIGN, "%s", _( "You" ) ); locations.emplace_back( player_character.pos() ); for( const npc &guy : g->all_npcs() ) { charmenu.addentry( charnum++, true, MENU_AUTOASSIGN, guy.name ); locations.emplace_back( guy.pos() ); } pointmenu_cb callback( locations ); charmenu.callback = &callback; charmenu.w_y_setup = 0; charmenu.query(); if( charmenu.ret < 0 || static_cast( charmenu.ret ) >= locations.size() ) { return; } const size_t index = charmenu.ret; // The NPC is also required for "Add mission", so has to be in this scope npc *np = g->critter_at( locations[index], false ); player &p = np ? *np->as_player() : *player_character.as_player(); uilist nmenu; if( np != nullptr ) { std::stringstream data; data << np->name << " " << ( np->male ? _( "Male" ) : _( "Female" ) ) << std::endl; data << np->myclass.obj().get_name() << "; " << npc_attitude_name( np->get_attitude() ) << "; " << ( np->get_faction() ? np->get_faction()->name : _( "no faction" ) ) << "; " << ( np->get_faction() ? np->get_faction()->currency->nname( 1 ) : _( "no currency" ) ) << "; " << "api: " << np->get_faction_ver() << std::endl; if( np->has_destination() ) { data << string_format( _( "Destination: %s %s" ), np->goal.to_string(), overmap_buffer.ter( np->goal )->get_name() ) << std::endl; } else { data << _( "No destination." ) << std::endl; } data << string_format( _( "Trust: %d" ), np->op_of_u.trust ) << " " << string_format( _( "Fear: %d" ), np->op_of_u.fear ) << " " << string_format( _( "Value: %d" ), np->op_of_u.value ) << " " << string_format( _( "Anger: %d" ), np->op_of_u.anger ) << " " << string_format( _( "Owed: %d" ), np->op_of_u.owed ) << std::endl; data << string_format( _( "Aggression: %d" ), static_cast( np->personality.aggression ) ) << " " << string_format( _( "Bravery: %d" ), static_cast( np->personality.bravery ) ) << " " << string_format( _( "Collector: %d" ), static_cast( np->personality.collector ) ) << " " << string_format( _( "Altruism: %d" ), static_cast( np->personality.altruism ) ) << std::endl; data << _( "Needs:" ) << std::endl; for( const auto &need : np->needs ) { data << need << std::endl; } data << string_format( _( "Total morale: %d" ), np->get_morale_level() ) << std::endl; nmenu.text = data.str(); } else { nmenu.text = _( "Player" ); } enum { D_DESC, D_SKILLS, D_PROF, D_STATS, D_ITEMS, D_DELETE_ITEMS, D_ITEM_WORN, D_HP, D_STAMINA, D_MORALE, D_PAIN, D_NEEDS, D_HEALTHY, D_STATUS, D_MISSION_ADD, D_MISSION_EDIT, D_TELE, D_MUTATE, D_CLASS, D_ATTITUDE, D_OPINION, D_ADD_EFFECT, D_ASTHMA }; nmenu.addentry( D_DESC, true, 'D', "%s", _( "Edit [D]escription - Name, Age, Height or Blood type" ) ); nmenu.addentry( D_SKILLS, true, 's', "%s", _( "Edit [s]kills" ) ); nmenu.addentry( D_PROF, true, 'P', "%s", _( "Edit [P]roficiencies" ) ); nmenu.addentry( D_STATS, true, 't', "%s", _( "Edit s[t]ats" ) ); nmenu.addentry( D_ITEMS, true, 'i', "%s", _( "Grant [i]tems" ) ); nmenu.addentry( D_DELETE_ITEMS, true, 'd', "%s", _( "[d]elete (all) items" ) ); nmenu.addentry( D_ITEM_WORN, true, 'w', "%s", _( "[w]ear/[w]ield an item from player's inventory" ) ); nmenu.addentry( D_HP, true, 'h', "%s", _( "Set [h]it points" ) ); nmenu.addentry( D_STAMINA, true, 'S', "%s", _( "Set [S]tamina" ) ); nmenu.addentry( D_MORALE, true, 'o', "%s", _( "Set m[o]rale" ) ); nmenu.addentry( D_PAIN, true, 'p', "%s", _( "Cause [p]ain" ) ); nmenu.addentry( D_HEALTHY, true, 'a', "%s", _( "Set he[a]lth" ) ); nmenu.addentry( D_NEEDS, true, 'n', "%s", _( "Set [n]eeds" ) ); nmenu.addentry( D_MUTATE, true, 'u', "%s", _( "M[u]tate" ) ); nmenu.addentry( D_STATUS, true, hotkey_for_action( ACTION_PL_INFO, /*maximum_modifier_count=*/1 ), "%s", _( "Status Window [@]" ) ); nmenu.addentry( D_TELE, true, 'e', "%s", _( "t[e]leport" ) ); nmenu.addentry( D_ADD_EFFECT, true, 'E', "%s", _( "Add an [E]ffect" ) ); nmenu.addentry( D_ASTHMA, true, 'k', "%s", _( "Cause asthma attac[k]" ) ); nmenu.addentry( D_MISSION_EDIT, true, 'M', "%s", _( "Edit [M]issions (WARNING: Unstable!)" ) ); if( p.is_npc() ) { nmenu.addentry( D_MISSION_ADD, true, 'm', "%s", _( "Add [m]ission" ) ); nmenu.addentry( D_CLASS, true, 'c', "%s", _( "Randomize with [c]lass" ) ); nmenu.addentry( D_ATTITUDE, true, 'A', "%s", _( "Set [A]ttitude" ) ); nmenu.addentry( D_OPINION, true, 'O', "%s", _( "Set [O]pinion" ) ); } nmenu.query(); switch( nmenu.ret ) { case D_SKILLS: wishskill( &p ); break; case D_STATS: { uilist smenu; smenu.addentry( 0, true, 'S', "%s: %d", _( "Maximum strength" ), p.str_max ); smenu.addentry( 1, true, 'D', "%s: %d", _( "Maximum dexterity" ), p.dex_max ); smenu.addentry( 2, true, 'I', "%s: %d", _( "Maximum intelligence" ), p.int_max ); smenu.addentry( 3, true, 'P', "%s: %d", _( "Maximum perception" ), p.per_max ); smenu.query(); int *bp_ptr = nullptr; switch( smenu.ret ) { case 0: bp_ptr = &p.str_max; break; case 1: bp_ptr = &p.dex_max; break; case 2: bp_ptr = &p.int_max; break; case 3: bp_ptr = &p.per_max; break; default: break; } if( bp_ptr != nullptr ) { int value; if( query_int( value, _( "Set the stat to? Currently: %d" ), *bp_ptr ) && value >= 0 ) { *bp_ptr = value; p.reset_stats(); } } } break; case D_PROF: wishproficiency( &p ); break; case D_ITEMS: wishitem( &p ); break; case D_DELETE_ITEMS: if( !query_yn( _( "Delete all items from the target?" ) ) ) { break; } for( auto &it : p.worn ) { it.on_takeoff( p ); } p.worn.clear(); p.inv->clear(); p.remove_weapon(); break; case D_ITEM_WORN: { item_location loc = game_menus::inv::titled_menu( player_character, _( "Make target equip" ) ); if( !loc ) { break; } item &to_wear = *loc; if( to_wear.is_armor() ) { p.on_item_wear( to_wear ); p.worn.push_back( to_wear ); } else if( !to_wear.is_null() ) { p.weapon = to_wear; get_event_bus().send( p.getID(), p.weapon.typeId() ); } } break; case D_HP: { const int torso_hp = p.get_part_hp_cur( bodypart_id( "torso" ) ); const int head_hp = p.get_part_hp_cur( bodypart_id( "head" ) ); const int arm_l_hp = p.get_part_hp_cur( bodypart_id( "arm_l" ) ); const int arm_r_hp = p.get_part_hp_cur( bodypart_id( "arm_r" ) ); const int leg_l_hp = p.get_part_hp_cur( bodypart_id( "leg_l" ) ); const int leg_r_hp = p.get_part_hp_cur( bodypart_id( "leg_r" ) ); uilist smenu; smenu.addentry( 0, true, 'q', "%s: %d", _( "Torso" ), torso_hp ); smenu.addentry( 1, true, 'w', "%s: %d", _( "Head" ), head_hp ); smenu.addentry( 2, true, 'a', "%s: %d", _( "Left arm" ), arm_l_hp ); smenu.addentry( 3, true, 's', "%s: %d", _( "Right arm" ), arm_r_hp ); smenu.addentry( 4, true, 'z', "%s: %d", _( "Left leg" ), leg_l_hp ); smenu.addentry( 5, true, 'x', "%s: %d", _( "Right leg" ), leg_r_hp ); smenu.addentry( 6, true, 'e', "%s: %d", _( "All" ), p.get_lowest_hp() ); smenu.query(); bodypart_str_id bp = bodypart_str_id( "no_a_real_part" ); int bp_ptr = -1; bool all_select = false; switch( smenu.ret ) { case 0: bp = body_part_torso; bp_ptr = torso_hp; break; case 1: bp = body_part_head; bp_ptr = head_hp; break; case 2: bp = body_part_arm_l; bp_ptr = arm_l_hp; break; case 3: bp = body_part_arm_r; bp_ptr = arm_r_hp; break; case 4: bp = body_part_leg_l; bp_ptr = leg_l_hp; break; case 5: bp = body_part_leg_r; bp_ptr = leg_r_hp; break; case 6: all_select = true; break; default: break; } if( bp.is_valid() ) { int value; if( query_int( value, _( "Set the hitpoints to? Currently: %d" ), bp_ptr ) && value >= 0 ) { p.set_part_hp_cur( bp.id(), value ); p.reset_stats(); } } else if( all_select ) { int value; if( query_int( value, _( "Set the hitpoints to? Currently: %d" ), p.get_lowest_hp() ) && value >= 0 ) { for( bodypart_id part_id : p.get_all_body_parts( get_body_part_flags::only_main ) ) { p.set_part_hp_cur( part_id, value ); } p.reset_stats(); } } } break; case D_STAMINA: int value; if( query_int( value, _( "Set stamina to? Current: %d. Max: %d." ), p.get_stamina(), p.get_stamina_max() ) ) { if( value >= 0 && value <= p.get_stamina_max() ) { p.set_stamina( value ); } else { add_msg( m_bad, _( "Target stamina value out of bounds!" ) ); } } break; case D_MORALE: { int current_morale_level = p.get_morale_level(); int value; if( query_int( value, _( "Set the morale to? Currently: %d" ), current_morale_level ) ) { int morale_level_delta = value - current_morale_level; p.add_morale( MORALE_PERM_DEBUG, morale_level_delta ); p.apply_persistent_morale(); } } break; case D_OPINION: { uilist smenu; smenu.addentry( 0, true, 'h', "%s: %d", _( "trust" ), np->op_of_u.trust ); smenu.addentry( 1, true, 's', "%s: %d", _( "fear" ), np->op_of_u.fear ); smenu.addentry( 2, true, 't', "%s: %d", _( "value" ), np->op_of_u.value ); smenu.addentry( 3, true, 'f', "%s: %d", _( "anger" ), np->op_of_u.anger ); smenu.addentry( 4, true, 'd', "%s: %d", _( "owed" ), np->op_of_u.owed ); smenu.query(); int value; switch( smenu.ret ) { case 0: if( query_int( value, _( "Set trust to? Currently: %d" ), np->op_of_u.trust ) ) { np->op_of_u.trust = value; } break; case 1: if( query_int( value, _( "Set fear to? Currently: %d" ), np->op_of_u.fear ) ) { np->op_of_u.fear = value; } break; case 2: if( query_int( value, _( "Set value to? Currently: %d" ), np->op_of_u.value ) ) { np->op_of_u.value = value; } break; case 3: if( query_int( value, _( "Set anger to? Currently: %d" ), np->op_of_u.anger ) ) { np->op_of_u.anger = value; } break; case 4: if( query_int( value, _( "Set owed to? Currently: %d" ), np->op_of_u.owed ) ) { np->op_of_u.owed = value; } break; } } break; case D_DESC: { uilist smenu; std::string current_bloodt = io::enum_to_string( p.my_blood_type ) + ( p.blood_rh_factor ? "+" : "-" ); smenu.text = _( "Select a value and press enter to change it." ); smenu.addentry( 0, true, 'n', "%s: %s", _( "Current name" ), p.get_name() ); smenu.addentry( 1, true, 'a', "%s: %d", _( "Current age" ), p.base_age() ); smenu.addentry( 2, true, 'h', "%s: %d", _( "Current height in cm" ), p.base_height() ); smenu.addentry( 3, true, 'b', "%s: %s", _( "Current blood type:" ), current_bloodt ); smenu.query(); switch( smenu.ret ) { case 0: { std::string filterstring = p.name; string_input_popup popup; popup .title( _( "Rename:" ) ) .width( 85 ) .edit( filterstring ); if( popup.confirmed() ) { p.name = filterstring; } } break; case 1: { string_input_popup popup; popup.title( _( "Enter age in years. Minimum 16, maximum 55" ) ) .text( string_format( "%d", p.base_age() ) ) .only_digits( true ); const int result = popup.query_int(); if( result != 0 ) { p.set_base_age( clamp( result, 16, 55 ) ); } } break; case 2: { string_input_popup popup; popup.title( _( "Enter height in centimeters. Minimum 145, maximum 200" ) ) .text( string_format( "%d", p.base_height() ) ) .only_digits( true ); const int result = popup.query_int(); if( result != 0 ) { p.set_base_height( clamp( result, 145, 200 ) ); } } break; case 3: { uilist btype; btype.text = _( "Select blood type" ); btype.addentry( static_cast( blood_type::blood_O ), true, '1', "O" ); btype.addentry( static_cast( blood_type::blood_A ), true, '2', "A" ); btype.addentry( static_cast( blood_type::blood_B ), true, '3', "B" ); btype.addentry( static_cast( blood_type::blood_AB ), true, '4', "AB" ); btype.query(); if( btype.ret < 0 ) { break; } uilist bfac; bfac.text = _( "Select Rh factor" ); bfac.addentry( 0, true, '-', _( "negative" ) ); bfac.addentry( 1, true, '+', _( "positive" ) ); bfac.query(); if( bfac.ret < 0 ) { break; } p.my_blood_type = static_cast( btype.ret ); p.blood_rh_factor = static_cast( bfac.ret ); break; } } } break; case D_PAIN: { int value; if( query_int( value, _( "Cause how much pain? pain: %d" ), p.get_pain() ) ) { p.mod_pain( value ); } } break; case D_NEEDS: { uilist smenu; smenu.addentry( 0, true, 'h', "%s: %d", _( "Hunger" ), p.get_hunger() ); smenu.addentry( 1, true, 's', "%s: %d", _( "Stored kCal" ), p.get_stored_kcal() ); smenu.addentry( 2, true, 't', "%s: %d", _( "Thirst" ), p.get_thirst() ); smenu.addentry( 3, true, 'f', "%s: %d", _( "Fatigue" ), p.get_fatigue() ); smenu.addentry( 4, true, 'd', "%s: %d", _( "Sleep Deprivation" ), p.get_sleep_deprivation() ); smenu.addentry( 5, true, 'a', _( "Reset all basic needs" ) ); const auto &vits = vitamin::all(); for( const auto &v : vits ) { smenu.addentry( -1, true, 0, "%s: %d", v.second.name(), p.vitamin_get( v.first ) ); } smenu.query(); int value; switch( smenu.ret ) { case 0: if( query_int( value, _( "Set hunger to? Currently: %d" ), p.get_hunger() ) ) { p.set_hunger( value ); } break; case 1: if( query_int( value, _( "Set stored kCal to? Currently: %d" ), p.get_stored_kcal() ) ) { p.set_stored_kcal( value ); } break; case 2: if( query_int( value, _( "Set thirst to? Currently: %d" ), p.get_thirst() ) ) { p.set_thirst( value ); } break; case 3: if( query_int( value, _( "Set fatigue to? Currently: %d" ), p.get_fatigue() ) ) { p.set_fatigue( value ); } break; case 4: if( query_int( value, _( "Set sleep deprivation to? Currently: %d" ), p.get_sleep_deprivation() ) ) { p.set_sleep_deprivation( value ); } break; case 5: p.initialize_stomach_contents(); p.set_hunger( 0 ); p.set_thirst( 0 ); p.set_fatigue( 0 ); p.set_sleep_deprivation( 0 ); p.set_stored_kcal( p.get_healthy_kcal() ); break; default: if( smenu.ret >= 6 && smenu.ret < static_cast( vits.size() + 6 ) ) { auto iter = std::next( vits.begin(), smenu.ret - 6 ); if( query_int( value, _( "Set %s to? Currently: %d" ), iter->second.name(), p.vitamin_get( iter->first ) ) ) { p.vitamin_set( iter->first, value ); } } } } break; case D_MUTATE: wishmutate( &p ); break; case D_HEALTHY: { uilist smenu; smenu.addentry( 0, true, 'h', "%s: %d", _( "Health" ), p.get_healthy() ); smenu.addentry( 1, true, 'm', "%s: %d", _( "Health modifier" ), p.get_healthy_mod() ); smenu.addentry( 2, true, 'r', "%s: %d", _( "Radiation" ), p.get_rad() ); smenu.query(); int value; switch( smenu.ret ) { case 0: if( query_int( value, _( "Set the value to? Currently: %d" ), p.get_healthy() ) ) { p.set_healthy( value ); } break; case 1: if( query_int( value, _( "Set the value to? Currently: %d" ), p.get_healthy_mod() ) ) { p.set_healthy_mod( value ); } break; case 2: if( query_int( value, _( "Set the value to? Currently: %d" ), p.get_rad() ) ) { p.set_rad( value ); } break; default: break; } } break; case D_STATUS: p.disp_info(); break; case D_MISSION_ADD: { uilist types; types.text = _( "Choose mission type" ); const auto all_missions = mission_type::get_all(); std::vector mts; for( size_t i = 0; i < all_missions.size(); i++ ) { types.addentry( i, true, -1, all_missions[i].tname() ); mts.push_back( &all_missions[i] ); } types.query(); if( types.ret >= 0 && types.ret < static_cast( mts.size() ) ) { np->add_new_mission( mission::reserve_new( mts[types.ret]->id, np->getID() ) ); } } break; case D_MISSION_EDIT: mission_debug::edit( p ); break; case D_TELE: { if( const cata::optional newpos = g->look_around() ) { p.setpos( *newpos ); if( p.is_player() ) { if( p.is_mounted() ) { p.mounted_creature->setpos( *newpos ); } g->update_map( player_character ); } } } break; case D_CLASS: { uilist classes; classes.text = _( "Choose new class" ); std::vector ids; size_t i = 0; for( const npc_class &cl : npc_class::get_all() ) { ids.push_back( cl.id ); classes.addentry( i, true, -1, cl.get_name() ); i++; } classes.query(); if( classes.ret < static_cast( ids.size() ) && classes.ret >= 0 ) { np->randomize( ids[classes.ret] ); } } break; case D_ATTITUDE: { uilist attitudes_ui; attitudes_ui.text = _( "Choose new attitude" ); std::vector attitudes; for( int i = NPCATT_NULL; i < NPCATT_END; i++ ) { npc_attitude att_id = static_cast( i ); std::string att_name = npc_attitude_name( att_id ); attitudes.push_back( att_id ); if( att_name == _( "Unknown attitude" ) ) { continue; } attitudes_ui.addentry( i, true, -1, att_name ); } attitudes_ui.query(); if( attitudes_ui.ret < static_cast( attitudes.size() ) && attitudes_ui.ret >= 0 ) { np->set_attitude( attitudes[attitudes_ui.ret] ); } } break; case D_ADD_EFFECT: { const auto text = string_input_popup() .title( _( "Choose an effect to add." ) ) .width( 20 ) .text( "" ) .only_digits( false ) .query_string(); efftype_id effect( text ); int intensity = 0; int seconds = 0; query_int( intensity, _( "What intensity?" ) ); query_int( seconds, _( "How many seconds?" ), 600 ); if( effect.is_valid() ) { p.add_effect( effect, time_duration::from_seconds( seconds ), false, intensity ); } else { add_msg( _( "Invalid effect" ) ); } break; } case D_ASTHMA: { p.set_mutation( trait_ASTHMA ); p.add_effect( effect_asthma, 10_minutes ); break; } } } static std::string mission_status_string( mission::mission_status status ) { static const std::map desc{ { { mission::mission_status::yet_to_start, translate_marker( "Yet to start" ) }, { mission::mission_status::in_progress, translate_marker( "In progress" ) }, { mission::mission_status::success, translate_marker( "Success" ) }, { mission::mission_status::failure, translate_marker( "Failure" ) } } }; const auto &iter = desc.find( status ); if( iter != desc.end() ) { return _( iter->second ); } return _( "Bugged" ); } std::string mission_debug::describe( const mission &m ) { std::stringstream data; data << _( "Type:" ) << m.type->id.str(); data << _( " Status:" ) << mission_status_string( m.status ); data << _( " ID:" ) << m.uid; data << _( " NPC ID:" ) << m.npc_id; data << _( " Target:" ) << m.target.to_string(); data << _( "Player ID:" ) << m.player_id; return data.str(); } static void add_header( uilist &mmenu, const std::string &str ) { if( !mmenu.entries.empty() ) { mmenu.addentry( -1, false, -1, "" ); } uilist_entry header( -1, false, -1, str, c_yellow, c_yellow ); header.force_color = true; mmenu.entries.push_back( header ); } void mission_debug::edit( player &who ) { if( who.is_player() ) { edit_player(); } else if( who.is_npc() ) { edit_npc( dynamic_cast( who ) ); } } void mission_debug::edit_npc( npc &who ) { dialogue_chatbin &bin = who.chatbin; std::vector all_missions; uilist mmenu; mmenu.text = _( "Select mission to edit" ); add_header( mmenu, _( "Currently assigned missions:" ) ); for( mission *m : bin.missions_assigned ) { mmenu.addentry( all_missions.size(), true, MENU_AUTOASSIGN, "%s", m->type->id.c_str() ); all_missions.emplace_back( m ); } add_header( mmenu, _( "Not assigned missions:" ) ); for( mission *m : bin.missions ) { mmenu.addentry( all_missions.size(), true, MENU_AUTOASSIGN, "%s", m->type->id.c_str() ); all_missions.emplace_back( m ); } mmenu.query(); if( mmenu.ret < 0 || mmenu.ret >= static_cast( all_missions.size() ) ) { return; } edit_mission( *all_missions[mmenu.ret] ); } void mission_debug::edit_player() { std::vector all_missions; uilist mmenu; mmenu.text = _( "Select mission to edit" ); avatar &player_character = get_avatar(); add_header( mmenu, _( "Active missions:" ) ); for( mission *m : player_character.active_missions ) { mmenu.addentry( all_missions.size(), true, MENU_AUTOASSIGN, "%s", m->type->id.c_str() ); all_missions.emplace_back( m ); } add_header( mmenu, _( "Completed missions:" ) ); for( mission *m : player_character.completed_missions ) { mmenu.addentry( all_missions.size(), true, MENU_AUTOASSIGN, "%s", m->type->id.c_str() ); all_missions.emplace_back( m ); } add_header( mmenu, _( "Failed missions:" ) ); for( mission *m : player_character.failed_missions ) { mmenu.addentry( all_missions.size(), true, MENU_AUTOASSIGN, "%s", m->type->id.c_str() ); all_missions.emplace_back( m ); } mmenu.query(); if( mmenu.ret < 0 || mmenu.ret >= static_cast( all_missions.size() ) ) { return; } edit_mission( *all_missions[mmenu.ret] ); } static bool remove_from_vec( std::vector &vec, mission *m ) { auto iter = std::remove( vec.begin(), vec.end(), m ); bool ret = iter != vec.end(); vec.erase( iter, vec.end() ); return ret; } void mission_debug::remove_mission( mission &m ) { avatar &player_character = get_avatar(); if( remove_from_vec( player_character.active_missions, &m ) ) { add_msg( _( "Removing from active_missions" ) ); } if( remove_from_vec( player_character.completed_missions, &m ) ) { add_msg( _( "Removing from completed_missions" ) ); } if( remove_from_vec( player_character.failed_missions, &m ) ) { add_msg( _( "Removing from failed_missions" ) ); } if( player_character.active_mission == &m ) { player_character.active_mission = nullptr; add_msg( _( "Unsetting active mission" ) ); } npc *giver = g->find_npc( m.npc_id ); if( giver != nullptr ) { if( remove_from_vec( giver->chatbin.missions_assigned, &m ) ) { add_msg( _( "Removing from %s missions_assigned" ), giver->name ); } if( remove_from_vec( giver->chatbin.missions, &m ) ) { add_msg( _( "Removing from %s missions" ), giver->name ); } } } void mission_debug::edit_mission( mission &m ) { uilist mmenu; mmenu.text = describe( m ); enum { M_FAIL, M_SUCCEED, M_REMOVE }; mmenu.addentry( M_FAIL, true, 'f', "%s", _( "Fail mission" ) ); mmenu.addentry( M_SUCCEED, true, 'c', "%s", _( "Mark as complete" ) ); mmenu.addentry( M_REMOVE, true, 'r', "%s", _( "Remove mission without proper cleanup" ) ); mmenu.query(); switch( mmenu.ret ) { case M_FAIL: m.fail(); break; case M_SUCCEED: m.status = mission::mission_status::success; break; case M_REMOVE: remove_mission( m ); break; } } void draw_benchmark( const int max_difference ) { // call the draw procedure as many times as possible in max_difference milliseconds auto start_tick = std::chrono::steady_clock::now(); auto end_tick = std::chrono::steady_clock::now(); int64_t difference = 0; int draw_counter = 0; static_popup popup; popup.on_top( true ).message( "%s", _( "Benchmark in progress…" ) ); while( true ) { end_tick = std::chrono::steady_clock::now(); difference = std::chrono::duration_cast( end_tick - start_tick ).count(); if( difference >= max_difference ) { break; } g->invalidate_main_ui_adaptor(); ui_manager::redraw_invalidated(); refresh_display(); draw_counter++; } DebugLog( D_INFO, DC_ALL ) << "Draw benchmark:\n" << "\n| USE_TILES | RENDERER | FRAMEBUFFER_ACCEL | USE_COLOR_MODULATED_TEXTURES | FPS |" << "\n|:---:|:---:|:---:|:---:|:---:|\n| " << get_option( "USE_TILES" ) << " | " << #if !defined(__ANDROID__) get_option( "RENDERER" ) << " | " << #else get_option( "SOFTWARE_RENDERING" ) << " | " << #endif get_option( "FRAMEBUFFER_ACCEL" ) << " | " << get_option( "USE_COLOR_MODULATED_TEXTURES" ) << " | " << static_cast( 1000.0 * draw_counter / static_cast( difference ) ) << " |\n"; add_msg( m_info, _( "Drew %d times in %.3f seconds. (%.3f fps average)" ), draw_counter, difference / 1000.0, 1000.0 * draw_counter / static_cast( difference ) ); } void debug() { bool debug_menu_has_hotkey = hotkey_for_action( ACTION_DEBUG, /*maximum_modifier_count=*/ -1, false ).has_value(); cata::optional action = debug_menu_uilist( debug_menu_has_hotkey ); // For the "cheaty" options, disable achievements when used achievements_tracker &achievements = get_achievements(); static const std::unordered_set non_cheaty_options = { debug_menu_index::SAVE_SCREENSHOT, debug_menu_index::GAME_REPORT, debug_menu_index::ENABLE_ACHIEVEMENTS, debug_menu_index::BENCHMARK, debug_menu_index::SHOW_MSG, }; bool should_disable_achievements = action && !non_cheaty_options.count( *action ); if( should_disable_achievements && achievements.is_enabled() ) { static const std::string query( translate_marker( "Using this will disable achievements. Proceed?" "\nThey can be reenabled in the 'game' section of the menu." ) ); if( query_yn( _( query ) ) ) { achievements.set_enabled( false ); } else { action = cata::nullopt; } } if( !action ) { return; } get_event_bus().send( *action ); avatar &player_character = get_avatar(); map &here = get_map(); tripoint abs_sub = here.get_abs_sub(); switch( *action ) { case debug_menu_index::WISH: debug_menu::wishitem( &player_character ); break; case debug_menu_index::SHORT_TELEPORT: debug_menu::teleport_short(); break; case debug_menu_index::LONG_TELEPORT: debug_menu::teleport_long(); break; case debug_menu_index::REVEAL_MAP: { auto &cur_om = g->get_cur_om(); for( int i = 0; i < OMAPX; i++ ) { for( int j = 0; j < OMAPY; j++ ) { for( int k = -OVERMAP_DEPTH; k <= OVERMAP_HEIGHT; k++ ) { cur_om.seen( { i, j, k } ) = true; } } } add_msg( m_good, _( "Current overmap revealed." ) ); } break; case debug_menu_index::SPAWN_NPC: { shared_ptr_fast temp = make_shared_fast(); temp->normalize(); temp->randomize(); temp->spawn_at_precise( abs_sub.xy(), player_character.pos() + point( -4, -4 ) ); overmap_buffer.insert_npc( temp ); temp->form_opinion( player_character ); temp->mission = NPC_MISSION_NULL; temp->add_new_mission( mission::reserve_random( ORIGIN_ANY_NPC, temp->global_omt_location(), temp->getID() ) ); std::string new_fac_id = "solo_"; new_fac_id += temp->name; // create a new "lone wolf" faction for this one NPC faction *new_solo_fac = g->faction_manager_ptr->add_new_faction( temp->name, faction_id( new_fac_id ), faction_id( "no_faction" ) ); temp->set_fac( new_solo_fac ? new_solo_fac->id : faction_id( "no_faction" ) ); g->load_npcs(); } break; case debug_menu_index::SPAWN_MON: debug_menu::wishmonster( cata::nullopt ); break; case debug_menu_index::GAME_STATE: { std::string mfus; std::vector> sorted; for( int f = 0; f < m_flag::MF_MAX; f++ ) { sorted.push_back( {static_cast( f ), MonsterGenerator::generator().m_flag_usage_stats[f]} ); } std::sort( sorted.begin(), sorted.end(), []( std::pair a, std::pair b ) { return a.second != b.second ? a.second > b.second : a.first < b.first; } ); popup( player_character.total_daily_calories_string() ); for( auto &m_flag_stat : sorted ) { mfus += string_format( "%s;%d\n", io::enum_to_string( m_flag_stat.first ), m_flag_stat.second ); } DebugLog( D_INFO, DC_ALL ) << "Monster flag usage statistics:\nFLAG;COUNT\n" << mfus; std::fill( MonsterGenerator::generator().m_flag_usage_stats.begin(), MonsterGenerator::generator().m_flag_usage_stats.end(), 0 ); popup_top( "Monster flag usage statistics were dumped to debug.log and cleared." ); std::string s = _( "Location %d:%d in %d:%d, %s\n" ); s += _( "Current turn: %d.\n" ); s += ngettext( "%d creature exists.\n", "%d creatures exist.\n", g->num_creatures() ); std::unordered_map creature_counts; for( Creature &critter : g->all_creatures() ) { std::string this_name = critter.get_name(); creature_counts[this_name]++; } if( !creature_counts.empty() ) { std::vector> creature_names_sorted; for( const std::pair &it : creature_counts ) { creature_names_sorted.emplace_back( it ); } std::stable_sort( creature_names_sorted.begin(), creature_names_sorted.end(), []( auto a, auto b ) { return a.second > b.second; } ); s += _( "\nSpecific creature type list:\n" ); for( const std::pair &crit_name : creature_names_sorted ) { s += string_format( "%i %s\n", crit_name.second, crit_name.first ); } } popup_top( s.c_str(), player_character.posx(), player_character.posy(), abs_sub.x, abs_sub.y, overmap_buffer.ter( player_character.global_omt_location() )->get_name(), to_turns( calendar::turn - calendar::turn_zero ), g->num_creatures() ); for( const npc &guy : g->all_npcs() ) { tripoint t = guy.global_sm_location(); add_msg( m_info, _( "%s: map ( %d:%d ) pos ( %d:%d )" ), guy.name, t.x, t.y, guy.posx(), guy.posy() ); } add_msg( m_info, _( "(you: %d:%d)" ), player_character.posx(), player_character.posy() ); std::string stom = _( "Stomach Contents: %d ml / %d ml kCal: %d, Water: %d ml" ); add_msg( m_info, stom.c_str(), units::to_milliliter( player_character.stomach.contains() ), units::to_milliliter( player_character.stomach.capacity( player_character ) ), player_character.stomach.get_calories(), units::to_milliliter( player_character.stomach.get_water() ), player_character.get_hunger() ); stom = _( "Guts Contents: %d ml / %d ml kCal: %d, Water: %d ml\nHunger: %d, Thirst: %d, kCal: %d / %d" ); add_msg( m_info, stom.c_str(), units::to_milliliter( player_character.guts.contains() ), units::to_milliliter( player_character.guts.capacity( player_character ) ), player_character.guts.get_calories(), units::to_milliliter( player_character.guts.get_water() ), player_character.get_hunger(), player_character.get_thirst(), player_character.get_stored_kcal(), player_character.get_healthy_kcal() ); add_msg( m_info, _( "Body Mass Index: %.0f\nBasal Metabolic Rate: %i" ), player_character.get_bmi(), player_character.get_bmr() ); add_msg( m_info, _( "Player activity level: %s" ), player_character.activity_level_str() ); if( get_option( "STATS_THROUGH_KILLS" ) ) { add_msg( m_info, _( "Kill xp: %d" ), player_character.kill_xp() ); } g->invalidate_main_ui_adaptor(); g->disp_NPCs(); break; } case debug_menu_index::KILL_NPCS: for( npc &guy : g->all_npcs() ) { add_msg( _( "%s's head implodes!" ), guy.name ); guy.set_part_hp_cur( bodypart_id( "head" ), 0 ); } break; case debug_menu_index::MUTATE: debug_menu::wishmutate( &player_character ); break; case debug_menu_index::SPAWN_VEHICLE: if( here.veh_at( player_character.pos() ) ) { dbg( D_ERROR ) << "game:load: There's already vehicle here"; debugmsg( "There's already vehicle here" ); } else { // Vector of name, id so that we can sort by name std::vector> veh_strings; for( auto &elem : vehicle_prototype::get_all() ) { if( elem == vproto_id( "custom" ) ) { continue; } veh_strings.emplace_back( elem->name.translated(), elem ); } std::sort( veh_strings.begin(), veh_strings.end(), localized_compare ); uilist veh_menu; veh_menu.text = _( "Choose vehicle to spawn" ); int menu_ind = 0; for( auto &elem : veh_strings ) { //~ Menu entry in vehicle wish menu: 1st string: displayed name, 2nd string: internal name of vehicle veh_menu.addentry( menu_ind, true, MENU_AUTOASSIGN, _( "%1$s (%2$s)" ), elem.first, elem.second.c_str() ); ++menu_ind; } veh_menu.query(); if( veh_menu.ret >= 0 && veh_menu.ret < static_cast( veh_strings.size() ) ) { // Didn't cancel const vproto_id &selected_opt = veh_strings[veh_menu.ret].second; tripoint dest = player_character.pos(); uilist veh_cond_menu; veh_cond_menu.text = _( "Vehicle condition" ); veh_cond_menu.addentry( 0, true, MENU_AUTOASSIGN, _( "Light damage" ) ); veh_cond_menu.addentry( 1, true, MENU_AUTOASSIGN, _( "Undamaged" ) ); veh_cond_menu.addentry( 2, true, MENU_AUTOASSIGN, _( "Disabled (tires or engine)" ) ); veh_cond_menu.query(); if( veh_cond_menu.ret >= 0 && veh_cond_menu.ret < 3 ) { // TODO: Allow picking this when add_vehicle has 3d argument vehicle *veh = here.add_vehicle( selected_opt, dest, -90_degrees, 100, veh_cond_menu.ret - 1 ); if( veh != nullptr ) { here.board_vehicle( dest, &player_character ); } } } } break; case debug_menu_index::CHANGE_SKILLS: { debug_menu::wishskill( &player_character ); } break; case debug_menu_index::LEARN_MA: add_msg( m_info, _( "Martial arts debug." ) ); add_msg( _( "Your eyes blink rapidly as knowledge floods your brain." ) ); for( auto &style : all_martialart_types() ) { if( style != matype_id( "style_none" ) ) { player_character.martial_arts_data->add_martialart( style ); } } add_msg( m_good, _( "You now know a lot more than just 10 styles of kung fu." ) ); break; case debug_menu_index::UNLOCK_RECIPES: { add_msg( m_info, _( "Recipe debug." ) ); add_msg( _( "Your eyes blink rapidly as knowledge floods your brain." ) ); for( const auto &e : recipe_dict ) { player_character.learn_recipe( &e.second ); } add_msg( m_good, _( "You know how to craft that now." ) ); } break; case debug_menu_index::EDIT_PLAYER: debug_menu::character_edit_menu(); break; case debug_menu_index::SPAWN_ARTIFACT: if( const cata::optional center = g->look_around() ) { artifact_natural_property prop = static_cast( rng( ARTPROP_NULL + 1, ARTPROP_MAX - 1 ) ); here.create_anomaly( *center, prop ); here.spawn_artifact( *center, relic_procgen_id( "alien_reality" ) ); } break; case debug_menu_index::SPAWN_CLAIRVOYANCE: player_character.i_add( item( "architect_cube", calendar::turn ) ); break; case debug_menu_index::MAP_EDITOR: g->look_debug(); break; case debug_menu_index::CHANGE_WEATHER: { uilist weather_menu; weather_menu.text = _( "Select new weather pattern:" ); weather_menu.addentry( 0, true, MENU_AUTOASSIGN, g->weather.weather_override == WEATHER_NULL ? _( "Keep normal weather patterns" ) : _( "Disable weather forcing" ) ); for( size_t i = 0; i < weather_types::get_all().size(); i++ ) { weather_menu.addentry( i, true, MENU_AUTOASSIGN, weather_types::get_all()[i].name.translated() ); } weather_menu.query(); if( weather_menu.ret >= 0 && static_cast( weather_menu.ret ) < weather_types::get_all().size() ) { const weather_type_id selected_weather = weather_types::get_all()[weather_menu.ret].id; g->weather.weather_override = selected_weather; g->weather.set_nextweather( calendar::turn ); } } break; case debug_menu_index::WIND_DIRECTION: { uilist wind_direction_menu; wind_direction_menu.text = _( "Select new wind direction:" ); wind_direction_menu.addentry( 0, true, MENU_AUTOASSIGN, g->weather.wind_direction_override ? _( "Disable direction forcing" ) : _( "Keep normal wind direction" ) ); int count = 1; for( int angle = 0; angle <= 315; angle += 45 ) { wind_direction_menu.addentry( count, true, MENU_AUTOASSIGN, get_wind_arrow( angle ) ); count += 1; } wind_direction_menu.query(); if( wind_direction_menu.ret == 0 ) { g->weather.wind_direction_override = cata::nullopt; } else if( wind_direction_menu.ret >= 0 && wind_direction_menu.ret < 9 ) { g->weather.wind_direction_override = ( wind_direction_menu.ret - 1 ) * 45; g->weather.set_nextweather( calendar::turn ); } } break; case debug_menu_index::WIND_SPEED: { uilist wind_speed_menu; wind_speed_menu.text = _( "Select new wind speed:" ); wind_speed_menu.addentry( 0, true, MENU_AUTOASSIGN, g->weather.wind_direction_override ? _( "Disable speed forcing" ) : _( "Keep normal wind speed" ) ); int count = 1; for( int speed = 0; speed <= 100; speed += 10 ) { std::string speedstring = std::to_string( speed ) + " " + velocity_units( VU_WIND ); wind_speed_menu.addentry( count, true, MENU_AUTOASSIGN, speedstring ); count += 1; } wind_speed_menu.query(); if( wind_speed_menu.ret == 0 ) { g->weather.windspeed_override = cata::nullopt; } else if( wind_speed_menu.ret >= 0 && wind_speed_menu.ret < 12 ) { int selected_wind_speed = ( wind_speed_menu.ret - 1 ) * 10; g->weather.windspeed_override = selected_wind_speed; g->weather.set_nextweather( calendar::turn ); } } break; case debug_menu_index::GEN_SOUND: { const cata::optional where = g->look_around(); if( !where ) { return; } int volume; if( !query_int( volume, _( "Volume of sound: " ) ) ) { return; } if( volume < 0 ) { return; } sounds::sound( *where, volume, sounds::sound_t::order, string_format( _( "DEBUG SOUND ( %d )" ), volume ) ); } break; case debug_menu_index::KILL_MONS: { for( monster &critter : g->all_monsters() ) { // Use the normal death functions, useful for testing death // and for getting a corpse. if( critter.type->id != mon_generator ) { critter.die( nullptr ); } } g->cleanup_dead(); } break; case debug_menu_index::DISPLAY_HORDES: ui::omap::display_hordes(); break; case debug_menu_index::TEST_IT_GROUP: { item_group::debug_spawn(); } break; // Damage Self case debug_menu_index::DAMAGE_SELF: { const int torso_hp = player_character.get_part_hp_cur( bodypart_id( "torso" ) ); const int head_hp = player_character.get_part_hp_cur( bodypart_id( "head" ) ); const int arm_l_hp = player_character.get_part_hp_cur( bodypart_id( "arm_l" ) ); const int arm_r_hp = player_character.get_part_hp_cur( bodypart_id( "arm_r" ) ); const int leg_l_hp = player_character.get_part_hp_cur( bodypart_id( "leg_l" ) ); const int leg_r_hp = player_character.get_part_hp_cur( bodypart_id( "leg_r" ) ); uilist smenu; smenu.addentry( 0, true, 'q', "%s: %d", _( "Torso" ), torso_hp ); smenu.addentry( 1, true, 'w', "%s: %d", _( "Head" ), head_hp ); smenu.addentry( 2, true, 'a', "%s: %d", _( "Left arm" ), arm_l_hp ); smenu.addentry( 3, true, 's', "%s: %d", _( "Right arm" ), arm_r_hp ); smenu.addentry( 4, true, 'z', "%s: %d", _( "Left leg" ), leg_l_hp ); smenu.addentry( 5, true, 'x', "%s: %d", _( "Right leg" ), leg_r_hp ); smenu.query(); bodypart_id part; int dbg_damage; switch( smenu.ret ) { case 0: part = bodypart_id( "torso" ); break; case 1: part = bodypart_id( "head" ); break; case 2: part = bodypart_id( "arm_l" ); break; case 3: part = bodypart_id( "arm_r" ); break; case 4: part = bodypart_id( "leg_l" ); break; case 5: part = bodypart_id( "leg_r" ); break; default: break; } if( query_int( dbg_damage, _( "Damage self for how much? hp: %s" ), part.id().c_str() ) ) { player_character.apply_damage( nullptr, part, dbg_damage ); player_character.die( nullptr ); } } break; // Add bleeding case debug_menu_index::BLEED_SELF: { uilist smenu; smenu.addentry( 0, true, 'q', _( "Torso" ) ); smenu.addentry( 1, true, 'w', _( "Head" ) ); smenu.addentry( 2, true, 'a', _( "Left arm" ) ); smenu.addentry( 3, true, 's', _( "Right arm" ) ); smenu.addentry( 4, true, 'z', _( "Left leg" ) ); smenu.addentry( 5, true, 'x', _( "Right leg" ) ); smenu.query(); bodypart_id part; int intensity = 0; switch( smenu.ret ) { case 0: part = bodypart_id( "torso" ); break; case 1: part = bodypart_id( "head" ); break; case 2: part = bodypart_id( "arm_l" ); break; case 3: part = bodypart_id( "arm_r" ); break; case 4: part = bodypart_id( "leg_l" ); break; case 5: part = bodypart_id( "leg_r" ); break; default: break; } if( query_int( intensity, _( "Add bleeding duration in minutes, equal to intensity:" ) ) ) { player_character.make_bleed( effect_source::empty(), part, 1_minutes * intensity ); } } break; case debug_menu_index::SHOW_SOUND: { #if defined(TILES) const auto &sounds_to_draw = sounds::get_monster_sounds(); shared_ptr_fast sound_cb = make_shared_fast( [&]() { const point offset { player_character.view_offset.xy() + point( POSX - player_character.posx(), POSY - player_character.posy() ) }; for( const auto &sound : sounds_to_draw.first ) { mvwputch( g->w_terrain, offset + sound.xy(), c_yellow, '?' ); } for( const auto &sound : sounds_to_draw.second ) { mvwputch( g->w_terrain, offset + sound.xy(), c_red, '?' ); } } ); g->add_draw_callback( sound_cb ); ui_manager::redraw(); inp_mngr.wait_for_any_key(); #else popup( _( "This binary was not compiled with tiles support." ) ); #endif } break; case debug_menu_index::DISPLAY_WEATHER: ui::omap::display_weather(); break; case debug_menu_index::DISPLAY_SCENTS: ui::omap::display_scents(); break; case debug_menu_index::DISPLAY_SCENTS_LOCAL: g->display_toggle_overlay( ACTION_DISPLAY_SCENT ); break; case debug_menu_index::DISPLAY_SCENTS_TYPE_LOCAL: g->display_toggle_overlay( ACTION_DISPLAY_SCENT_TYPE ); break; case debug_menu_index::DISPLAY_TEMP: g->display_toggle_overlay( ACTION_DISPLAY_TEMPERATURE ); break; case debug_menu_index::DISPLAY_VEHICLE_AI: g->display_toggle_overlay( ACTION_DISPLAY_VEHICLE_AI ); break; case debug_menu_index::DISPLAY_VISIBILITY: g->display_toggle_overlay( ACTION_DISPLAY_VISIBILITY ); break; case debug_menu_index::DISPLAY_LIGHTING: g->display_toggle_overlay( ACTION_DISPLAY_LIGHTING ); break; case debug_menu_index::DISPLAY_RADIATION: g->display_toggle_overlay( ACTION_DISPLAY_RADIATION ); break; case debug_menu_index::DISPLAY_TRANSPARENCY: g->display_toggle_overlay( ACTION_DISPLAY_TRANSPARENCY ); break; case debug_menu_index::DISPLAY_REACHABILITY_ZONES: g->display_reachability_zones(); break; case debug_menu_index::HOUR_TIMER: g->toggle_debug_hour_timer(); break; case debug_menu_index::CHANGE_TIME: { auto set_turn = [&]( const int initial, const time_duration & factor, const char *const msg ) { const auto text = string_input_popup() .title( msg ) .width( 20 ) .text( std::to_string( initial ) ) .only_digits( true ) .query_string(); if( text.empty() ) { return; } const int new_value = std::atoi( text.c_str() ); const time_duration offset = ( new_value - initial ) * factor; // Arbitrary maximal value. const time_point max = calendar::turn_zero + time_duration::from_turns( std::numeric_limits::max() / 2 ); calendar::turn = std::max( std::min( max, calendar::turn + offset ), calendar::turn_zero ); }; uilist smenu; static const auto years = []( const time_point & p ) { return static_cast( ( p - calendar::turn_zero ) / calendar::year_length() ); }; do { const int iSel = smenu.ret; smenu.reset(); smenu.addentry( 0, true, 'y', "%s: %d", _( "year" ), years( calendar::turn ) ); smenu.addentry( 1, !calendar::eternal_season(), 's', "%s: %d", _( "season" ), static_cast( season_of_year( calendar::turn ) ) ); smenu.addentry( 2, true, 'd', "%s: %d", _( "day" ), day_of_season( calendar::turn ) ); smenu.addentry( 3, true, 'h', "%s: %d", _( "hour" ), hour_of_day( calendar::turn ) ); smenu.addentry( 4, true, 'm', "%s: %d", _( "minute" ), minute_of_hour( calendar::turn ) ); smenu.addentry( 5, true, 't', "%s: %d", _( "turn" ), to_turns( calendar::turn - calendar::turn_zero ) ); smenu.selected = iSel; smenu.query(); switch( smenu.ret ) { case 0: set_turn( years( calendar::turn ), calendar::year_length(), _( "Set year to?" ) ); break; case 1: set_turn( static_cast( season_of_year( calendar::turn ) ), calendar::season_length(), _( "Set season to? (0 = spring)" ) ); break; case 2: set_turn( day_of_season( calendar::turn ), 1_days, _( "Set days to?" ) ); break; case 3: set_turn( hour_of_day( calendar::turn ), 1_hours, _( "Set hour to?" ) ); break; case 4: set_turn( minute_of_hour( calendar::turn ), 1_minutes, _( "Set minute to?" ) ); break; case 5: set_turn( to_turns( calendar::turn - calendar::turn_zero ), 1_turns, string_format( _( "Set turn to? (One day is %i turns)" ), to_turns( 1_days ) ).c_str() ); break; default: break; } } while( smenu.ret != UILIST_CANCEL ); } break; case debug_menu_index::SET_AUTOMOVE: { const cata::optional dest = g->look_around(); if( !dest || *dest == player_character.pos() ) { break; } auto rt = here.route( player_character.pos(), *dest, player_character.get_pathfinding_settings(), player_character.get_path_avoid() ); if( !rt.empty() ) { player_character.set_destination( rt ); } else { popup( "Couldn't find path" ); } } break; case debug_menu_index::SHOW_MUT_CAT: for( const auto &elem : player_character.mutation_category_level ) { add_msg( "%s: %d", elem.first.c_str(), elem.second ); } break; case debug_menu_index::OM_EDITOR: ui::omap::display_editor(); break; case debug_menu_index::BENCHMARK: { const int ms = string_input_popup() .title( _( "Enter benchmark length (in milliseconds):" ) ) .width( 20 ) .text( "5000" ) .query_int(); debug_menu::draw_benchmark( ms ); } break; case debug_menu_index::OM_TELEPORT: debug_menu::teleport_overmap(); break; case debug_menu_index::OM_TELEPORT_COORDINATES: debug_menu::teleport_overmap( true ); break; case debug_menu_index::TRAIT_GROUP: trait_group::debug_spawn(); break; case debug_menu_index::ENABLE_ACHIEVEMENTS: if( achievements.is_enabled() ) { popup( _( "Achievements are already enabled" ) ); } else { achievements.set_enabled( true ); popup( _( "Achievements enabled" ) ); } break; case debug_menu_index::SHOW_MSG: debugmsg( "Test debugmsg" ); break; case debug_menu_index::CRASH_GAME: raise( SIGSEGV ); break; case debug_menu_index::MAP_EXTRA: { const std::vector &mx_str = MapExtras::get_all_function_names(); uilist mx_menu; for( const std::string &extra : mx_str ) { mx_menu.addentry( -1, true, -1, extra ); } mx_menu.query(); int mx_choice = mx_menu.ret; if( mx_choice >= 0 && mx_choice < static_cast( mx_str.size() ) ) { const tripoint_abs_omt where_omt( ui::omap::choose_point() ); if( where_omt != overmap::invalid_tripoint ) { tripoint_abs_sm where_sm = project_to( where_omt ); tinymap mx_map; mx_map.load( where_sm, false ); MapExtras::apply_function( mx_str[mx_choice], mx_map, where_sm.raw() ); g->load_npcs(); here.invalidate_map_cache( here.get_abs_sub().z ); } } break; } case debug_menu_index::NESTED_MAPGEN: debug_menu::spawn_nested_mapgen(); break; case debug_menu_index::DISPLAY_NPC_PATH: g->debug_pathfinding = !g->debug_pathfinding; break; case debug_menu_index::PRINT_FACTION_INFO: { int count = 0; for( const auto &elem : g->faction_manager_ptr->all() ) { std::cout << std::to_string( count ) << " Faction_id key in factions map = " << elem.first.str() << std::endl; std::cout << std::to_string( count ) << " Faction name associated with this id is " << elem.second.name << std::endl; std::cout << std::to_string( count ) << " the id of that faction object is " << elem.second.id.str() << std::endl; count++; } std::cout << "Player faction is " << player_character.get_faction()->id.str() << std::endl; break; } case debug_menu_index::PRINT_NPC_MAGIC: { for( npc &guy : g->all_npcs() ) { const std::vector spells = guy.magic->spells(); if( spells.empty() ) { std::cout << guy.disp_name() << " does not know any spells." << std::endl; continue; } std::cout << guy.disp_name() << "knows : "; int counter = 1; for( const spell_id &sp : spells ) { std::cout << sp->name.translated() << " "; if( counter < static_cast( spells.size() ) ) { std::cout << "and "; } else { std::cout << "." << std::endl; } counter++; } } break; } case debug_menu_index::QUIT_NOSAVE: if( query_yn( _( "Quit without saving? This may cause issues such as duplicated or missing items and vehicles!" ) ) ) { player_character.moves = 0; g->uquit = QUIT_NOSAVED; } break; case debug_menu_index::TEST_WEATHER: { get_weather().get_cur_weather_gen().test_weather( g->get_seed(), get_weather().next_instance_allowed ); } break; case debug_menu_index::SAVE_SCREENSHOT: { #if defined(TILES) // check that the current '/screenshots' directory exists std::stringstream map_directory; map_directory << PATH_INFO::world_base_save_path() << "/screenshots/"; assure_dir_exist( map_directory.str() ); // build file name: /screenshots/[]_.png // Date format is a somewhat ISO-8601 compliant GMT time date (except for some characters that wouldn't pass on most file systems like ':'). std::time_t time = std::time( nullptr ); std::stringstream date_buffer; date_buffer << std::put_time( std::gmtime( &time ), "%F_%H-%M-%S_%z" ); const auto tmp_file_name = string_format( "[%s]_%s.png", player_character.get_name(), date_buffer.str() ); std::string file_name = ensure_valid_file_name( tmp_file_name ); auto current_file_path = map_directory.str() + file_name; // Take a screenshot of the viewport. if( g->take_screenshot( current_file_path ) ) { popup( _( "Successfully saved your screenshot to: %s" ), map_directory.str() ); } else { popup( _( "An error occurred while trying to save the screenshot." ) ); } #else popup( _( "This binary was not compiled with tiles support." ) ); #endif } break; case debug_menu_index::GAME_REPORT: { // generate a game report, useful for bug reporting. std::string report = game_info::game_report(); // write to log DebugLog( DL_ALL, DC_ALL ) << " GAME REPORT:\n" << report; std::string popup_msg = _( "Report written to debug.log" ); #if defined(TILES) // copy to clipboard int clipboard_result = SDL_SetClipboardText( report.c_str() ); printErrorIf( clipboard_result != 0, "Error while copying the game report to the clipboard." ); if( clipboard_result == 0 ) { popup_msg += _( " and to the clipboard." ); } #endif popup( popup_msg ); } break; case debug_menu_index::LEARN_SPELLS: if( spell_type::get_all().empty() ) { add_msg( m_bad, _( "There are no spells to learn. You must install a mod that adds some." ) ); } else { for( const spell_type &learn : spell_type::get_all() ) { player_character.magic->learn_spell( &learn, player_character, true ); } add_msg( m_good, _( "You have become an Archwizardpriest! What will you do with your newfound power?" ) ); } break; case debug_menu_index::LEVEL_SPELLS: { std::vector spells = player_character.magic->get_spells(); if( spells.empty() ) { add_msg( m_bad, _( "Try learning some spells first." ) ); return; } std::vector uiles; { uilist_entry uile( _( "Spell" ) ); uile.ctxt = string_format( "%s %s", //~ translation should not exceed 4 console cells right_justify( _( "LVL" ), 4 ), //~ translation should not exceed 4 console cells right_justify( _( "MAX" ), 4 ) ); uile.enabled = false; uiles.emplace_back( uile ); } int retval = 0; for( spell *sp : spells ) { uilist_entry uile( sp->name() ); uile.ctxt = string_format( "%4d %4d", sp->get_level(), sp->get_max_level() ); uile.retval = retval++; uile.enabled = !sp->is_max_level(); uiles.emplace_back( uile ); } int action = uilist( _( "Debug level spell:" ), uiles ); if( action < 0 ) { return; } int desired_level = 0; int cur_level = spells[action]->get_level(); query_int( desired_level, _( "Desired Spell Level: (Current %d)" ), cur_level ); desired_level = std::min( desired_level, spells[action]->get_max_level() ); while( cur_level < desired_level ) { spells[action]->gain_level(); cur_level = spells[action]->get_level(); } add_msg( m_good, _( "%s is now level %d!" ), spells[action]->name(), spells[action]->get_level() ); break; } case debug_menu_index::TEST_MAP_EXTRA_DISTRIBUTION: MapExtras::debug_spawn_test(); break; case debug_menu_index::GENERATE_EFFECT_LIST: write_to_file( "effect_list.output", [&]( std::ostream & testfile ) { testfile << "|;id;duration;intensity;perm;bp" << std::endl; for( const effect &eff : get_player_character().get_effects() ) { testfile << "|;" << eff.get_id().str() << ";" << to_string( eff.get_duration() ) << ";" << eff.get_intensity() << ";" << ( eff.is_permanent() ? "true" : "false" ) << ";" << eff.get_bp().id().str() << std::endl; } }, "effect_list" ); popup( _( "Effect list written to effect_list.output" ) ); break; case debug_menu_index::VEHICLE_BATTERY_CHARGE: { optional_vpart_position v_part_pos = here.veh_at( player_character.pos() ); if( !v_part_pos ) { add_msg( m_bad, _( "There's no vehicle there." ) ); break; } int amount = 0; string_input_popup popup; popup .title( _( "By how much? (in kJ, negative to discharge)" ) ) .width( 30 ) .edit( amount ); if( !popup.canceled() ) { vehicle &veh = v_part_pos->vehicle(); if( amount >= 0 ) { veh.charge_battery( amount, false ); } else { veh.discharge_battery( -amount, false ); } } break; } case debug_menu_index::last: return; } here.invalidate_map_cache( here.get_abs_sub().z ); } } // namespace debug_menu