X-Git-Url: http://git.osdn.net/view?a=blobdiff_plain;f=src%2Ffloor%2Ffloor-save.c;h=95d63a067d15283b16d70d091d1592dc73ef16bd;hb=ca29fa7d4433bea71a02152ebe67edbbb01a8974;hp=f6ea5f26de9255a8f57d4a5a50253cf864e36f21;hpb=a24aa4fefa93e293499c1d9bfe1da4c7b337934a;p=hengband%2Fhengband.git diff --git a/src/floor/floor-save.c b/src/floor/floor-save.c index f6ea5f26d..95d63a067 100644 --- a/src/floor/floor-save.c +++ b/src/floor/floor-save.c @@ -10,82 +10,37 @@ */ #include "floor/floor-save.h" -#include "action/travel-execution.h" -#include "cmd-building/cmd-building.h" -#include "cmd-io/cmd-dump.h" #include "core/asking-player.h" -#include "dungeon/dungeon.h" -#include "dungeon/quest.h" -#include "floor/cave.h" -#include "floor/floor-events.h" -#include "floor/floor-generator.h" -#include "floor/floor-object.h" -#include "floor/floor.h" -#include "floor/geometry.h" -#include "floor/wild.h" -#include "game-option/birth-options.h" -#include "game-option/play-record-options.h" -#include "grid/feature.h" -#include "grid/grid.h" -#include "inventory/inventory-slot-types.h" +#include "floor/floor-save-util.h" #include "io/files-util.h" #include "io/uid-checker.h" -#include "io/write-diary.h" -#include "load/floor-loader.h" -#include "main/sound-of-music.h" -#include "mind/mind-mirror-master.h" -#include "mind/mind-ninja.h" -#include "monster-floor/monster-generator.h" -#include "monster-floor/monster-remover.h" -#include "monster-floor/monster-summon.h" #include "monster-race/monster-race.h" -#include "monster-race/race-flags1.h" -#include "monster-race/race-flags2.h" -#include "monster-race/race-flags7.h" -#include "monster/monster-describer.h" -#include "monster/monster-description-types.h" -#include "monster/monster-flag-types.h" #include "monster/monster-info.h" -#include "monster/monster-list.h" #include "monster/monster-status.h" -#include "monster/monster-update.h" -#include "monster/smart-learn-types.h" -#include "object-hook/hook-checker.h" -#include "object-hook/hook-enchant.h" -#include "pet/pet-util.h" -#include "player/player-class.h" -#include "player/player-personalities-types.h" -#include "player/special-defense-types.h" -#include "save/floor-writer.h" -#include "spell-kind/spells-floor.h" -#include "system/artifact-type-definition.h" #include "system/floor-type-definition.h" -#include "system/system-variables.h" +#include "system/monster-type-definition.h" #include "util/angband-files.h" -#include "util/bit-flags-calculator.h" #include "view/display-messages.h" -#include "window/main-window-util.h" -#include "world/world.h" -bool repair_monsters; - -static FLOOR_IDX new_floor_id; /*!<次のフロアのID / floor_id of the destination */ -static u32b latest_visit_mark; /*!<フロアを渡った回数?(確認中) / Max number of visit_mark */ -#define MAX_PARTY_MON 21 /*!< フロア移動時に先のフロアに連れて行けるペットの最大数 Maximum number of preservable pets */ -static monster_type party_mon[MAX_PARTY_MON]; /*!< フロア移動に保存するペットモンスターの配列 */ - -/* - * Number of floor_id used from birth - */ -FLOOR_IDX max_floor_id; +static void check_saved_tmp_files(const int fd, bool *force) +{ + if (fd >= 0) { + (void)fd_close(fd); + return; + } + + if (*force) + return; -/* - * Sign for current process used in temporary files. - * Actually it is the start time of current process. - */ -u32b saved_floor_file_sign; + msg_print(_("エラー:古いテンポラリ・ファイルが残っています。", "Error: There are old temporary files.")); + msg_print(_("変愚蛮怒を二重に起動していないか確認してください。", "Make sure you are not running two game processes simultaneously.")); + msg_print(_("過去に変愚蛮怒がクラッシュした場合は一時ファイルを", "If the temporary files are garbage from an old crashed process, ")); + msg_print(_("強制的に削除して実行を続けられます。", "you can delete them safely.")); + if (!get_check(_("強制的に削除してもよろしいですか?", "Do you delete the old temporary files? "))) + quit(_("実行中止", "Aborted.")); -saved_floor_type saved_floors[MAX_SAVED_FLOORS]; + *force = TRUE; +} /*! * @brief 保存フロア配列を初期化する / Initialize saved_floors array. @@ -97,65 +52,25 @@ saved_floor_type saved_floors[MAX_SAVED_FLOORS]; void init_saved_floors(player_type *creature_ptr, bool force) { char floor_savefile[1024]; - int i; int fd = -1; BIT_FLAGS mode = 0644; - for (i = 0; i < MAX_SAVED_FLOORS; i++) { + for (int i = 0; i < MAX_SAVED_FLOORS; i++) { saved_floor_type *sf_ptr = &saved_floors[i]; - - /* File name */ sprintf(floor_savefile, "%s.F%02d", savefile, i); - - /* Grab permissions */ safe_setuid_grab(creature_ptr); - - /* Try to create the file */ fd = fd_make(floor_savefile, mode); - - /* Drop permissions */ safe_setuid_drop(); - - /* Failed! */ - if (fd < 0) { - if (!force) { - msg_print(_("エラー:古いテンポラリ・ファイルが残っています。", "Error: There are old temporary files.")); - msg_print(_("変愚蛮怒を二重に起動していないか確認してください。", "Make sure you are not running two game processes simultaneously.")); - msg_print(_("過去に変愚蛮怒がクラッシュした場合は一時ファイルを", "If the temporary files are garbage from an old crashed process, ")); - msg_print(_("強制的に削除して実行を続けられます。", "you can delete them safely.")); - if (!get_check(_("強制的に削除してもよろしいですか?", "Do you delete the old temporary files? "))) - quit(_("実行中止", "Aborted.")); - force = TRUE; - } - } else { - /* Close the "fd" */ - (void)fd_close(fd); - } - - /* Grab permissions */ + check_saved_tmp_files(fd, &force); safe_setuid_grab(creature_ptr); - - /* Simply kill the temporary file */ (void)fd_kill(floor_savefile); - - /* Drop permissions */ safe_setuid_drop(); - sf_ptr->floor_id = 0; } - /* No floor_id used yet (No.0 is reserved to indicate non existance) */ max_floor_id = 1; - - /* vist_mark is from 1 */ latest_visit_mark = 1; - - /* A sign to mark temporary files */ saved_floor_file_sign = (u32b)time(NULL); - - /* No next floor yet */ new_floor_id = 0; - - /* No change floor mode yet */ creature_ptr->change_floor_mode = 0; } @@ -168,26 +83,14 @@ void init_saved_floors(player_type *creature_ptr, bool force) void clear_saved_floor_files(player_type *creature_ptr) { char floor_savefile[1024]; - int i; - for (i = 0; i < MAX_SAVED_FLOORS; i++) { + for (int i = 0; i < MAX_SAVED_FLOORS; i++) { saved_floor_type *sf_ptr = &saved_floors[i]; - - /* No temporary file */ - if (!sf_ptr->floor_id) - continue; - if (sf_ptr->floor_id == creature_ptr->floor_id) + if ((sf_ptr->floor_id == 0) || (sf_ptr->floor_id == creature_ptr->floor_id)) continue; - /* File name */ sprintf(floor_savefile, "%s.F%02d", savefile, i); - - /* Grab permissions */ safe_setuid_grab(creature_ptr); - - /* Simply kill the temporary file */ (void)fd_kill(floor_savefile); - - /* Drop permissions */ safe_setuid_drop(); } } @@ -199,20 +102,15 @@ void clear_saved_floor_files(player_type *creature_ptr) */ saved_floor_type *get_sf_ptr(FLOOR_IDX floor_id) { - int i; - - /* floor_id No.0 indicates no floor */ if (!floor_id) return NULL; - for (i = 0; i < MAX_SAVED_FLOORS; i++) { + for (int i = 0; i < MAX_SAVED_FLOORS; i++) { saved_floor_type *sf_ptr = &saved_floors[i]; - if (sf_ptr->floor_id == floor_id) return sf_ptr; } - /* None found */ return NULL; } @@ -222,37 +120,44 @@ saved_floor_type *get_sf_ptr(FLOOR_IDX floor_id) * @param sf_ptr 保存フロアの参照ポインタ * @return なし */ -static void kill_saved_floor(player_type *creature_ptr, saved_floor_type *sf_ptr) +void kill_saved_floor(player_type *creature_ptr, saved_floor_type *sf_ptr) { char floor_savefile[1024]; - if (!sf_ptr) - return; - - /* Already empty */ - if (!sf_ptr->floor_id) + if (!sf_ptr || (sf_ptr->floor_id == 0)) return; if (sf_ptr->floor_id == creature_ptr->floor_id) { - /* Kill current floor */ creature_ptr->floor_id = 0; + sf_ptr->floor_id = 0; + return; + } - /* Current floor doesn't have temporary file */ - } else { - /* File name */ - sprintf(floor_savefile, "%s.F%02d", savefile, (int)sf_ptr->savefile_id); + sprintf(floor_savefile, "%s.F%02d", savefile, (int)sf_ptr->savefile_id); + safe_setuid_grab(creature_ptr); + (void)fd_kill(floor_savefile); + safe_setuid_drop(); + sf_ptr->floor_id = 0; +} - /* Grab permissions */ - safe_setuid_grab(creature_ptr); +static void find_oldest_floor_id(player_type *creature_ptr, saved_floor_type *sf_ptr, FLOOR_IDX *fl_idx) +{ + if (*fl_idx != MAX_SAVED_FLOORS) + return; - /* Simply kill the temporary file */ - (void)fd_kill(floor_savefile); + s16b oldest = 0; + u32b oldest_visit = 0xffffffffL; + for (*fl_idx = 0; *fl_idx < MAX_SAVED_FLOORS; (*fl_idx)++) { + sf_ptr = &saved_floors[*fl_idx]; + if ((sf_ptr->floor_id == creature_ptr->floor_id) || (sf_ptr->visit_mark > oldest_visit)) + continue; - /* Drop permissions */ - safe_setuid_drop(); + oldest = *fl_idx; + oldest_visit = sf_ptr->visit_mark; } - /* No longer exists */ - sf_ptr->floor_id = 0; + sf_ptr = &saved_floors[oldest]; + kill_saved_floor(creature_ptr, sf_ptr); + *fl_idx = oldest; } /*! @@ -265,236 +170,30 @@ static void kill_saved_floor(player_type *creature_ptr, saved_floor_type *sf_ptr FLOOR_IDX get_new_floor_id(player_type *creature_ptr) { saved_floor_type *sf_ptr = NULL; - FLOOR_IDX i; - - /* Look for empty space */ - for (i = 0; i < MAX_SAVED_FLOORS; i++) { - sf_ptr = &saved_floors[i]; - + FLOOR_IDX fl_idx; + for (fl_idx = 0; fl_idx < MAX_SAVED_FLOORS; fl_idx++) { + sf_ptr = &saved_floors[fl_idx]; if (!sf_ptr->floor_id) break; } - /* None found */ - if (i == MAX_SAVED_FLOORS) { - s16b oldest = 0; - u32b oldest_visit = 0xffffffffL; - - /* Search for oldest */ - for (i = 0; i < MAX_SAVED_FLOORS; i++) { - sf_ptr = &saved_floors[i]; - - /* Don't kill current floor */ - if (sf_ptr->floor_id == creature_ptr->floor_id) - continue; - - /* Don't kill newer */ - if (sf_ptr->visit_mark > oldest_visit) - continue; - - oldest = i; - oldest_visit = sf_ptr->visit_mark; - } - - /* Kill oldest saved floor */ - sf_ptr = &saved_floors[oldest]; - kill_saved_floor(creature_ptr, sf_ptr); - - /* Use it */ - i = oldest; - } - - /* Prepare new floor data */ - sf_ptr->savefile_id = i; + find_oldest_floor_id(creature_ptr, sf_ptr, &fl_idx); + sf_ptr->savefile_id = fl_idx; sf_ptr->floor_id = max_floor_id; sf_ptr->last_visit = 0; sf_ptr->upper_floor_id = 0; sf_ptr->lower_floor_id = 0; sf_ptr->visit_mark = latest_visit_mark++; - - /* sf_ptr->dun_level may be changed later */ sf_ptr->dun_level = creature_ptr->current_floor_ptr->dun_level; - - /* Increment number of floor_id */ if (max_floor_id < MAX_SHORT) max_floor_id++; - - /* 32767 floor_ids are all used up! Re-use ancient IDs */ else - max_floor_id = 1; + max_floor_id = 1; // 32767 floor_ids are all used up! Re-use ancient IDs. return sf_ptr->floor_id; } /*! - * @brief フロア切り替え時の処理フラグを追加する / Prepare mode flags of changing floor - * @param creature_ptr プレーヤーへの参照ポインタ - * @param mode 追加したい所持フラグ - * @return なし - */ -void prepare_change_floor_mode(player_type *creature_ptr, BIT_FLAGS mode) { creature_ptr->change_floor_mode |= mode; } - -/*! - * @brief 階段移動先のフロアが生成できない時に簡単な行き止まりマップを作成する / Builds the dead end - * @return なし - */ -static void build_dead_end(player_type *creature_ptr) -{ - POSITION x, y; - - clear_cave(creature_ptr); - - /* Mega-Hack -- no player yet */ - creature_ptr->x = creature_ptr->y = 0; - - /* Fill the arrays of floors and walls in the good proportions */ - set_floor_and_wall(0); - - /* Smallest area */ - creature_ptr->current_floor_ptr->height = SCREEN_HGT; - creature_ptr->current_floor_ptr->width = SCREEN_WID; - - /* Filled with permanent walls */ - for (y = 0; y < MAX_HGT; y++) { - for (x = 0; x < MAX_WID; x++) { - /* Create "solid" perma-wall */ - place_bold(creature_ptr, y, x, GB_SOLID_PERM); - } - } - - /* Place at center of the floor */ - creature_ptr->y = creature_ptr->current_floor_ptr->height / 2; - creature_ptr->x = creature_ptr->current_floor_ptr->width / 2; - - /* Give one square */ - place_bold(creature_ptr, creature_ptr->y, creature_ptr->x, GB_FLOOR); - - wipe_generate_random_floor_flags(creature_ptr->current_floor_ptr); -} - -/*! - * @brief フロア移動時のペット保存処理 / Preserve_pets - * @param master_ptr プレーヤーへの参照ポインタ - * @return なし - */ -static void preserve_pet(player_type *master_ptr) -{ - int num; - MONSTER_IDX i; - - for (num = 0; num < MAX_PARTY_MON; num++) { - party_mon[num].r_idx = 0; - } - - if (master_ptr->riding) { - monster_type *m_ptr = &master_ptr->current_floor_ptr->m_list[master_ptr->riding]; - - /* Pet of other pet don't follow. */ - if (m_ptr->parent_m_idx) { - master_ptr->riding = 0; - master_ptr->pet_extra_flags &= ~(PF_TWO_HANDS); - master_ptr->riding_ryoute = master_ptr->old_riding_ryoute = FALSE; - } else { - /* Preserve the mount */ - (void)COPY(&party_mon[0], m_ptr, monster_type); - - /* Delete from this floor */ - delete_monster_idx(master_ptr, master_ptr->riding); - } - } - - /* - * If player is in wild mode, no pets are preserved - * except a monster whom player riding - */ - if (!master_ptr->wild_mode && !master_ptr->current_floor_ptr->inside_arena && !master_ptr->phase_out) { - for (i = master_ptr->current_floor_ptr->m_max - 1, num = 1; (i >= 1 && num < MAX_PARTY_MON); i--) { - monster_type *m_ptr = &master_ptr->current_floor_ptr->m_list[i]; - - if (!monster_is_valid(m_ptr)) - continue; - if (!is_pet(m_ptr)) - continue; - if (i == master_ptr->riding) - continue; - - if (reinit_wilderness) { - /* Don't lose sight of pets when getting a Quest */ - } else { - POSITION dis = distance(master_ptr->y, master_ptr->x, m_ptr->fy, m_ptr->fx); - - /* Confused (etc.) monsters don't follow. */ - if (monster_confused_remaining(m_ptr) || monster_stunned_remaining(m_ptr) || monster_csleep_remaining(m_ptr)) - continue; - - /* Pet of other pet don't follow. */ - if (m_ptr->parent_m_idx) - continue; - - /* - * Pets with nickname will follow even from 3 blocks away - * when you or the pet can see the other. - */ - if (m_ptr->nickname - && ((player_has_los_bold(master_ptr, m_ptr->fy, m_ptr->fx) && projectable(master_ptr, master_ptr->y, master_ptr->x, m_ptr->fy, m_ptr->fx)) - || (los(master_ptr, m_ptr->fy, m_ptr->fx, master_ptr->y, master_ptr->x) - && projectable(master_ptr, m_ptr->fy, m_ptr->fx, master_ptr->y, master_ptr->x)))) { - if (dis > 3) - continue; - } else { - if (dis > 1) - continue; - } - } - - (void)COPY(&party_mon[num], &master_ptr->current_floor_ptr->m_list[i], monster_type); - - num++; - - /* Delete from this floor */ - delete_monster_idx(master_ptr, i); - } - } - - if (record_named_pet) { - for (i = master_ptr->current_floor_ptr->m_max - 1; i >= 1; i--) { - monster_type *m_ptr = &master_ptr->current_floor_ptr->m_list[i]; - GAME_TEXT m_name[MAX_NLEN]; - - if (!monster_is_valid(m_ptr)) - continue; - if (!is_pet(m_ptr)) - continue; - if (!m_ptr->nickname) - continue; - if (master_ptr->riding == i) - continue; - - monster_desc(master_ptr, m_name, m_ptr, MD_ASSUME_VISIBLE | MD_INDEF_VISIBLE); - exe_write_diary(master_ptr, DIARY_NAMED_PET, RECORD_NAMED_PET_MOVED, m_name); - } - } - - /* Pet of other pet may disappear. */ - for (i = master_ptr->current_floor_ptr->m_max - 1; i >= 1; i--) { - monster_type *m_ptr = &master_ptr->current_floor_ptr->m_list[i]; - - /* Are there its parent? */ - if (m_ptr->parent_m_idx && !master_ptr->current_floor_ptr->m_list[m_ptr->parent_m_idx].r_idx) { - /* Its parent have gone, it also goes away. */ - - if (is_seen(master_ptr, m_ptr)) { - GAME_TEXT m_name[MAX_NLEN]; - monster_desc(master_ptr, m_name, m_ptr, 0); - msg_format(_("%sは消え去った!", "%^s disappears!"), m_name); - } - - delete_monster_idx(master_ptr, i); - } - } -} - -/*! * @brief フロア移動時にペットを伴った場合の準備処理 / Pre-calculate the racial counters of preserved pets * @param master_ptr プレーヤーへの参照ポインタ * @return なし @@ -504,812 +203,12 @@ static void preserve_pet(player_type *master_ptr) void precalc_cur_num_of_pet(player_type *player_ptr) { monster_type *m_ptr; - int i; int max_num = player_ptr->wild_mode ? 1 : MAX_PARTY_MON; - - for (i = 0; i < max_num; i++) { + for (int i = 0; i < max_num; i++) { m_ptr = &party_mon[i]; - - /* Skip empty monsters */ if (!monster_is_valid(m_ptr)) continue; - /* Hack -- Increase the racial counter */ real_r_ptr(m_ptr)->cur_num++; } } - -/*! - * @brief 移動先のフロアに伴ったペットを配置する / Place preserved pet monsters on new floor - * @param master_ptr プレーヤーへの参照ポインタ - * @return なし - */ -static void place_pet(player_type *master_ptr) -{ - int i; - int max_num = master_ptr->wild_mode ? 1 : MAX_PARTY_MON; - floor_type *floor_ptr = master_ptr->current_floor_ptr; - for (i = 0; i < max_num; i++) { - POSITION cy = 0, cx = 0; - MONSTER_IDX m_idx; - - if (!(party_mon[i].r_idx)) - continue; - - if (i == 0) { - m_idx = m_pop(floor_ptr); - master_ptr->riding = m_idx; - if (m_idx) { - cy = master_ptr->y; - cx = master_ptr->x; - } - } else { - int j; - POSITION d; - - for (d = 1; d < A_MAX; d++) { - for (j = 1000; j > 0; j--) { - scatter(master_ptr, &cy, &cx, master_ptr->y, master_ptr->x, d, 0); - if (monster_can_enter(master_ptr, cy, cx, &r_info[party_mon[i].r_idx], 0)) - break; - } - if (j) - break; - } - m_idx = (d == 6) ? 0 : m_pop(floor_ptr); - } - - if (m_idx) { - monster_type *m_ptr = &master_ptr->current_floor_ptr->m_list[m_idx]; - monster_race *r_ptr; - - master_ptr->current_floor_ptr->grid_array[cy][cx].m_idx = m_idx; - - m_ptr->r_idx = party_mon[i].r_idx; - - /* Copy all member of the structure */ - *m_ptr = party_mon[i]; - r_ptr = real_r_ptr(m_ptr); - - m_ptr->fy = cy; - m_ptr->fx = cx; - m_ptr->current_floor_ptr = master_ptr->current_floor_ptr; - m_ptr->ml = TRUE; - m_ptr->mtimed[MTIMED_CSLEEP] = 0; - m_ptr->hold_o_idx = 0; - m_ptr->target_y = 0; - - if ((r_ptr->flags1 & RF1_FORCE_SLEEP) && !ironman_nightmare) { - /* Monster is still being nice */ - m_ptr->mflag |= (MFLAG_NICE); - - /* Must repair monsters */ - repair_monsters = TRUE; - } - update_monster(master_ptr, m_idx, TRUE); - lite_spot(master_ptr, cy, cx); - - /* Pre-calculated in precalc_cur_num_of_pet() */ - /* r_ptr->cur_num++; */ - - /* Hack -- Count the number of "reproducers" */ - if (r_ptr->flags2 & RF2_MULTIPLY) - master_ptr->current_floor_ptr->num_repro++; - - } else { - monster_type *m_ptr = &party_mon[i]; - monster_race *r_ptr = real_r_ptr(m_ptr); - GAME_TEXT m_name[MAX_NLEN]; - - monster_desc(master_ptr, m_name, m_ptr, 0); - msg_format(_("%sとはぐれてしまった。", "You have lost sight of %s."), m_name); - if (record_named_pet && m_ptr->nickname) { - monster_desc(master_ptr, m_name, m_ptr, MD_INDEF_VISIBLE); - exe_write_diary(master_ptr, DIARY_NAMED_PET, RECORD_NAMED_PET_LOST_SIGHT, m_name); - } - - /* Pre-calculated in precalc_cur_num_of_pet(), but need to decrease */ - if (r_ptr->cur_num) - r_ptr->cur_num--; - } - } - - /* For accuracy of precalc_cur_num_of_pet() */ - (void)C_WIPE(party_mon, MAX_PARTY_MON, monster_type); -} - -/*! - * @brief ユニークモンスターやアーティファクトの所在フロアを更新する / Hack -- Update location of unique monsters and artifacts - * @param cur_floor_id 現在のフロアID - * @return なし - * @details - * The r_ptr->floor_id and a_ptr->floor_id are not updated correctly\n - * while new floor creation since dungeons may be re-created by\n - * auto-scum option.\n - */ -static void update_unique_artifact(floor_type *floor_ptr, s16b cur_floor_id) -{ - int i; - - /* Maintain unique monsters */ - for (i = 1; i < floor_ptr->m_max; i++) { - monster_race *r_ptr; - monster_type *m_ptr = &floor_ptr->m_list[i]; - - if (!monster_is_valid(m_ptr)) - continue; - - /* Extract real monster race */ - r_ptr = real_r_ptr(m_ptr); - - /* Memorize location of the unique monster */ - if ((r_ptr->flags1 & RF1_UNIQUE) || (r_ptr->flags7 & RF7_NAZGUL)) { - r_ptr->floor_id = cur_floor_id; - } - } - - /* Maintain artifatcs */ - for (i = 1; i < floor_ptr->o_max; i++) { - object_type *o_ptr = &floor_ptr->o_list[i]; - - if (!object_is_valid(o_ptr)) - continue; - - /* Memorize location of the artifact */ - if (object_is_fixed_artifact(o_ptr)) { - a_info[o_ptr->name1].floor_id = cur_floor_id; - } - } -} - -/*! - * @brief フロア移動時、プレイヤーの移動先モンスターが既にいた場合ランダムな近隣に移動させる / When a monster is at a place where player will return, - * @return なし - */ -static void get_out_monster(player_type *protected_ptr) -{ - int tries = 0; - POSITION dis = 1; - POSITION oy = protected_ptr->y; - POSITION ox = protected_ptr->x; - floor_type *floor_ptr = protected_ptr->current_floor_ptr; - MONSTER_IDX m_idx = floor_ptr->grid_array[oy][ox].m_idx; - - /* Nothing to do if no monster */ - if (!m_idx) - return; - - /* Look until done */ - while (TRUE) { - monster_type *m_ptr; - - /* Pick a (possibly illegal) location */ - POSITION ny = rand_spread(oy, dis); - POSITION nx = rand_spread(ox, dis); - - tries++; - - /* Stop after 1000 tries */ - if (tries > 10000) - return; - - /* - * Increase distance after doing enough tries - * compared to area of possible space - */ - if (tries > 20 * dis * dis) - dis++; - - /* Ignore illegal locations */ - if (!in_bounds(floor_ptr, ny, nx)) - continue; - - /* Require "empty" floor space */ - if (!is_cave_empty_bold(protected_ptr, ny, nx)) - continue; - - /* Hack -- no teleport onto glyph of warding */ - if (is_glyph_grid(&floor_ptr->grid_array[ny][nx])) - continue; - if (is_explosive_rune_grid(&floor_ptr->grid_array[ny][nx])) - continue; - - /* ...nor onto the Pattern */ - if (pattern_tile(floor_ptr, ny, nx)) - continue; - - /*** It's a good place ***/ - - m_ptr = &floor_ptr->m_list[m_idx]; - - /* Update the old location */ - floor_ptr->grid_array[oy][ox].m_idx = 0; - - /* Update the new location */ - floor_ptr->grid_array[ny][nx].m_idx = m_idx; - - /* Move the monster */ - m_ptr->fy = ny; - m_ptr->fx = nx; - - /* No need to do update_monster() */ - - /* Success */ - return; - } -} - -/*! - * @brief 新フロアに移動元フロアに繋がる階段を配置する / Virtually teleport onto the stairs that is connecting between two floors. - * @param sf_ptr 移動元の保存フロア構造体参照ポインタ - * @return なし - */ -static void locate_connected_stairs(player_type *creature_ptr, floor_type *floor_ptr, saved_floor_type *sf_ptr, BIT_FLAGS floor_mode) -{ - POSITION x, y, sx = 0, sy = 0; - POSITION x_table[20]; - POSITION y_table[20]; - int num = 0; - int i; - - /* Search usable stairs */ - for (y = 0; y < floor_ptr->height; y++) { - for (x = 0; x < floor_ptr->width; x++) { - grid_type *g_ptr = &floor_ptr->grid_array[y][x]; - feature_type *f_ptr = &f_info[g_ptr->feat]; - bool ok = FALSE; - - if (floor_mode & CFM_UP) { - if (have_flag(f_ptr->flags, FF_LESS) && have_flag(f_ptr->flags, FF_STAIRS) && !have_flag(f_ptr->flags, FF_SPECIAL)) { - ok = TRUE; - - /* Found fixed stairs? */ - if (g_ptr->special && g_ptr->special == sf_ptr->upper_floor_id) { - sx = x; - sy = y; - } - } - } - - else if (floor_mode & CFM_DOWN) { - if (have_flag(f_ptr->flags, FF_MORE) && have_flag(f_ptr->flags, FF_STAIRS) && !have_flag(f_ptr->flags, FF_SPECIAL)) { - ok = TRUE; - - /* Found fixed stairs */ - if (g_ptr->special && g_ptr->special == sf_ptr->lower_floor_id) { - sx = x; - sy = y; - } - } - } - - else { - if (have_flag(f_ptr->flags, FF_BLDG)) { - ok = TRUE; - } - } - - if (ok && (num < 20)) { - x_table[num] = x; - y_table[num] = y; - num++; - } - } - } - - if (sx) { - /* Already fixed */ - creature_ptr->y = sy; - creature_ptr->x = sx; - } else if (!num) { - /* No stairs found! -- No return */ - prepare_change_floor_mode(creature_ptr, CFM_RAND_PLACE | CFM_NO_RETURN); - - /* Mega Hack -- It's not the stairs you enter. Disable it. */ - if (!feat_uses_special(floor_ptr->grid_array[creature_ptr->y][creature_ptr->x].feat)) - floor_ptr->grid_array[creature_ptr->y][creature_ptr->x].special = 0; - } else { - /* Choose random one */ - i = randint0(num); - - /* Point stair location */ - creature_ptr->y = y_table[i]; - creature_ptr->x = x_table[i]; - } -} - -/*! - * @brief 現在のフロアを離れるに伴って行なわれる保存処理 - * / Maintain quest monsters, mark next floor_id at stairs, save current floor, and prepare to enter next floor. - * @param creature_ptr プレーヤーへの参照ポインタ - * @return なし - */ -void leave_floor(player_type *creature_ptr) -{ - grid_type *g_ptr = NULL; - feature_type *f_ptr; - saved_floor_type *sf_ptr; - MONRACE_IDX quest_r_idx = 0; - DUNGEON_IDX i; - FLOOR_IDX tmp_floor_idx = 0; - - /* Preserve pets and prepare to take these to next floor */ - preserve_pet(creature_ptr); - - /* Remove all mirrors without explosion */ - remove_all_mirrors(creature_ptr, FALSE); - - if (creature_ptr->special_defense & NINJA_S_STEALTH) - set_superstealth(creature_ptr, FALSE); - - /* New floor is not yet prepared */ - new_floor_id = 0; - - /* Temporarily get a floor_id (for Arena) */ - if (!creature_ptr->floor_id && (creature_ptr->change_floor_mode & CFM_SAVE_FLOORS) && !(creature_ptr->change_floor_mode & CFM_NO_RETURN)) { - /* Get temporary floor_id */ - tmp_floor_idx = get_new_floor_id(creature_ptr); - } - - /* Search the quest monster index */ - for (i = 0; i < max_q_idx; i++) { - if ((quest[i].status == QUEST_STATUS_TAKEN) && ((quest[i].type == QUEST_TYPE_KILL_LEVEL) || (quest[i].type == QUEST_TYPE_RANDOM)) - && (quest[i].level == creature_ptr->current_floor_ptr->dun_level) && (creature_ptr->dungeon_idx == quest[i].dungeon) - && !(quest[i].flags & QUEST_FLAG_PRESET)) { - quest_r_idx = quest[i].r_idx; - } - } - - /* Maintain quest monsters */ - for (i = 1; i < creature_ptr->current_floor_ptr->m_max; i++) { - monster_race *r_ptr; - monster_type *m_ptr = &creature_ptr->current_floor_ptr->m_list[i]; - - if (!monster_is_valid(m_ptr)) - continue; - - /* Only maintain quest monsters */ - if (quest_r_idx != m_ptr->r_idx) - continue; - - /* Extract real monster race */ - r_ptr = real_r_ptr(m_ptr); - - /* Ignore unique monsters */ - if ((r_ptr->flags1 & RF1_UNIQUE) || (r_ptr->flags7 & RF7_NAZGUL)) - continue; - - /* Delete non-unique quest monsters */ - delete_monster_idx(creature_ptr, i); - } - - /* Check if there is a same item */ - for (i = 0; i < INVEN_PACK; i++) { - object_type *o_ptr = &creature_ptr->inventory_list[i]; - - if (!object_is_valid(o_ptr)) - continue; - - /* Delete old memorized location of the artifact */ - if (object_is_fixed_artifact(o_ptr)) { - a_info[o_ptr->name1].floor_id = 0; - } - } - - /* Extract current floor info or NULL */ - sf_ptr = get_sf_ptr(creature_ptr->floor_id); - - /* Choose random stairs */ - if ((creature_ptr->change_floor_mode & CFM_RAND_CONNECT) && tmp_floor_idx) { - locate_connected_stairs(creature_ptr, creature_ptr->current_floor_ptr, sf_ptr, creature_ptr->change_floor_mode); - } - - /* Extract new dungeon level */ - if (creature_ptr->change_floor_mode & CFM_SAVE_FLOORS) { - /* Extract stair position */ - g_ptr = &creature_ptr->current_floor_ptr->grid_array[creature_ptr->y][creature_ptr->x]; - f_ptr = &f_info[g_ptr->feat]; - - /* Get back to old saved floor? */ - if (g_ptr->special && !have_flag(f_ptr->flags, FF_SPECIAL) && get_sf_ptr(g_ptr->special)) { - /* Saved floor is exist. Use it. */ - new_floor_id = g_ptr->special; - } - - /* Mark shaft up/down */ - if (have_flag(f_ptr->flags, FF_STAIRS) && have_flag(f_ptr->flags, FF_SHAFT)) { - prepare_change_floor_mode(creature_ptr, CFM_SHAFT); - } - } - - /* Climb up/down some sort of stairs */ - if (creature_ptr->change_floor_mode & (CFM_DOWN | CFM_UP)) { - int move_num = 0; - - /* Extract level movement number */ - if (creature_ptr->change_floor_mode & CFM_DOWN) - move_num = 1; - else if (creature_ptr->change_floor_mode & CFM_UP) - move_num = -1; - - /* Shafts are deeper than normal stairs */ - if (creature_ptr->change_floor_mode & CFM_SHAFT) - move_num += SGN(move_num); - - /* Get out from or Enter the dungeon */ - if (creature_ptr->change_floor_mode & CFM_DOWN) { - if (!creature_ptr->current_floor_ptr->dun_level) - move_num = d_info[creature_ptr->dungeon_idx].mindepth; - } else if (creature_ptr->change_floor_mode & CFM_UP) { - if (creature_ptr->current_floor_ptr->dun_level + move_num < d_info[creature_ptr->dungeon_idx].mindepth) - move_num = -creature_ptr->current_floor_ptr->dun_level; - } - - creature_ptr->current_floor_ptr->dun_level += move_num; - } - - /* Leaving the dungeon to town */ - if (!creature_ptr->current_floor_ptr->dun_level && creature_ptr->dungeon_idx) { - creature_ptr->leaving_dungeon = TRUE; - if (!vanilla_town && !lite_town) { - creature_ptr->wilderness_y = d_info[creature_ptr->dungeon_idx].dy; - creature_ptr->wilderness_x = d_info[creature_ptr->dungeon_idx].dx; - } - creature_ptr->recall_dungeon = creature_ptr->dungeon_idx; - creature_ptr->dungeon_idx = 0; - - /* Reach to the surface -- Clear all saved floors */ - creature_ptr->change_floor_mode &= ~CFM_SAVE_FLOORS; // TODO - } - - /* Kill some old saved floors */ - if (!(creature_ptr->change_floor_mode & CFM_SAVE_FLOORS)) { - /* Kill all saved floors */ - for (i = 0; i < MAX_SAVED_FLOORS; i++) - kill_saved_floor(creature_ptr, &saved_floors[i]); - - /* Reset visit_mark count */ - latest_visit_mark = 1; - } else if (creature_ptr->change_floor_mode & CFM_NO_RETURN) { - /* Kill current floor */ - kill_saved_floor(creature_ptr, sf_ptr); - } - - /* No current floor -- Left/Enter dungeon etc... */ - if (!creature_ptr->floor_id) { - /* No longer need to save current floor */ - return; - } - - /* Mark next floor_id on the previous floor */ - if (!new_floor_id) { - /* Get new id */ - new_floor_id = get_new_floor_id(creature_ptr); - - /* Connect from here */ - if (g_ptr && !feat_uses_special(g_ptr->feat)) { - g_ptr->special = new_floor_id; - } - } - - /* Fix connection -- level teleportation or trap door */ - if (creature_ptr->change_floor_mode & CFM_RAND_CONNECT) { - if (creature_ptr->change_floor_mode & CFM_UP) - sf_ptr->upper_floor_id = new_floor_id; - else if (creature_ptr->change_floor_mode & CFM_DOWN) - sf_ptr->lower_floor_id = new_floor_id; - } - - /* If you can return, you need to save previous floor */ - if ((creature_ptr->change_floor_mode & CFM_SAVE_FLOORS) && !(creature_ptr->change_floor_mode & CFM_NO_RETURN)) { - /* Get out of the my way! */ - get_out_monster(creature_ptr); - - /* Record the last visit turn of current floor */ - sf_ptr->last_visit = current_world_ptr->game_turn; - - forget_lite(creature_ptr->current_floor_ptr); - forget_view(creature_ptr->current_floor_ptr); - clear_mon_lite(creature_ptr->current_floor_ptr); - - /* Save current floor */ - if (!save_floor(creature_ptr, sf_ptr, 0)) { - /* Save failed -- No return */ - prepare_change_floor_mode(creature_ptr, CFM_NO_RETURN); - - /* Kill current floor */ - kill_saved_floor(creature_ptr, get_sf_ptr(creature_ptr->floor_id)); - } - } -} - -/*! - * @brief フロアの切り替え処理 / Enter new floor. - * @param creature_ptr プレーヤーへの参照ポインタ - * @return なし - * @details - * If the floor is an old saved floor, it will be\n - * restored from the temporary file. If the floor is new one, new floor\n - * will be generated.\n - */ -void change_floor(player_type *creature_ptr) -{ - saved_floor_type *sf_ptr; - bool loaded = FALSE; - - /* The dungeon is not ready */ - current_world_ptr->character_dungeon = FALSE; - - /* No longer in the trap detecteded region */ - creature_ptr->dtrap = FALSE; - - /* Mega-Hack -- no panel yet */ - panel_row_min = 0; - panel_row_max = 0; - panel_col_min = 0; - panel_col_max = 0; - - /* Mega-Hack -- not ambushed on the wildness? */ - creature_ptr->ambush_flag = FALSE; - - /* No saved floors (On the surface etc.) */ - if (!(creature_ptr->change_floor_mode & CFM_SAVE_FLOORS) && !(creature_ptr->change_floor_mode & CFM_FIRST_FLOOR)) { - generate_floor(creature_ptr); - - /* Paranoia -- No new saved floor */ - new_floor_id = 0; - } - - /* In the dungeon */ - else { - /* No floor_id yet */ - if (!new_floor_id) { - /* Get new id */ - new_floor_id = get_new_floor_id(creature_ptr); - } - - /* Pointer for infomations of new floor */ - sf_ptr = get_sf_ptr(new_floor_id); - - /* Try to restore old floor */ - if (sf_ptr->last_visit) { - /* Old saved floor is exist */ - if (load_floor(creature_ptr, sf_ptr, 0)) { - loaded = TRUE; - - /* Forbid return stairs */ - if (creature_ptr->change_floor_mode & CFM_NO_RETURN) { - grid_type *g_ptr = &creature_ptr->current_floor_ptr->grid_array[creature_ptr->y][creature_ptr->x]; - - if (!feat_uses_special(g_ptr->feat)) { - if (creature_ptr->change_floor_mode & (CFM_DOWN | CFM_UP)) { - /* Reset to floor */ - g_ptr->feat = feat_ground_type[randint0(100)]; - } - - g_ptr->special = 0; - } - } - } - } - - /* - * Set lower/upper_floor_id of new floor when the new - * floor is right-above/right-under the current floor. - * - * Stair creation/Teleport level/Trap door will take - * you the same floor when you used it later again. - */ - if (creature_ptr->floor_id) { - saved_floor_type *cur_sf_ptr = get_sf_ptr(creature_ptr->floor_id); - - if (creature_ptr->change_floor_mode & CFM_UP) { - /* New floor is right-above */ - if (cur_sf_ptr->upper_floor_id == new_floor_id) - sf_ptr->lower_floor_id = creature_ptr->floor_id; - } else if (creature_ptr->change_floor_mode & CFM_DOWN) { - /* New floor is right-under */ - if (cur_sf_ptr->lower_floor_id == new_floor_id) - sf_ptr->upper_floor_id = creature_ptr->floor_id; - } - } - - /* Break connection to killed floor */ - else { - if (creature_ptr->change_floor_mode & CFM_UP) - sf_ptr->lower_floor_id = 0; - else if (creature_ptr->change_floor_mode & CFM_DOWN) - sf_ptr->upper_floor_id = 0; - } - - /* Maintain monsters and artifacts */ - if (loaded) { - MONSTER_IDX i; - GAME_TURN tmp_last_visit = sf_ptr->last_visit; - GAME_TURN absence_ticks; - int alloc_chance = d_info[creature_ptr->dungeon_idx].max_m_alloc_chance; - GAME_TURN alloc_times; - - while (tmp_last_visit > current_world_ptr->game_turn) - tmp_last_visit -= TURNS_PER_TICK * TOWN_DAWN; - absence_ticks = (current_world_ptr->game_turn - tmp_last_visit) / TURNS_PER_TICK; - - /* Maintain monsters */ - for (i = 1; i < creature_ptr->current_floor_ptr->m_max; i++) { - monster_race *r_ptr; - monster_type *m_ptr = &creature_ptr->current_floor_ptr->m_list[i]; - - if (!monster_is_valid(m_ptr)) - continue; - - if (!is_pet(m_ptr)) { - /* Restore HP */ - m_ptr->hp = m_ptr->maxhp = m_ptr->max_maxhp; - - /* Remove timed status (except MTIMED_CSLEEP) */ - (void)set_monster_fast(creature_ptr, i, 0); - (void)set_monster_slow(creature_ptr, i, 0); - (void)set_monster_stunned(creature_ptr, i, 0); - (void)set_monster_confused(creature_ptr, i, 0); - (void)set_monster_monfear(creature_ptr, i, 0); - (void)set_monster_invulner(creature_ptr, i, 0, FALSE); - } - - /* Extract real monster race */ - r_ptr = real_r_ptr(m_ptr); - - /* Ignore non-unique */ - if (!(r_ptr->flags1 & RF1_UNIQUE) && !(r_ptr->flags7 & RF7_NAZGUL)) - continue; - - /* Appear at a different floor? */ - if (r_ptr->floor_id != new_floor_id) { - /* Disapper from here */ - delete_monster_idx(creature_ptr, i); - } - } - - /* Maintain artifatcs */ - for (i = 1; i < creature_ptr->current_floor_ptr->o_max; i++) { - object_type *o_ptr = &creature_ptr->current_floor_ptr->o_list[i]; - - if (!object_is_valid(o_ptr)) - continue; - - /* Ignore non-artifact */ - if (!object_is_fixed_artifact(o_ptr)) - continue; - - /* Appear at a different floor? */ - if (a_info[o_ptr->name1].floor_id != new_floor_id) { - /* Disappear from here */ - delete_object_idx(creature_ptr, i); - } else { - /* Cancel preserve */ - a_info[o_ptr->name1].cur_num = 1; - } - } - - (void)place_quest_monsters(creature_ptr); - - /* Place some random monsters */ - alloc_times = absence_ticks / alloc_chance; - - if (randint0(alloc_chance) < (absence_ticks % alloc_chance)) - alloc_times++; - - for (i = 0; i < alloc_times; i++) { - /* Make a (group of) new monster */ - (void)alloc_monster(creature_ptr, 0, 0, summon_specific); - } - - } - - /* New floor_id or failed to restore */ - else /* if (!loaded) */ - { - if (sf_ptr->last_visit) { - /* Temporary file is broken? */ - msg_print(_("階段は行き止まりだった。", "The staircases come to a dead end...")); - - /* Create simple dead end */ - build_dead_end(creature_ptr); - - /* Break connection */ - if (creature_ptr->change_floor_mode & CFM_UP) { - sf_ptr->upper_floor_id = 0; - } else if (creature_ptr->change_floor_mode & CFM_DOWN) { - sf_ptr->lower_floor_id = 0; - } - } else { - generate_floor(creature_ptr); - } - - /* Record last visit turn */ - sf_ptr->last_visit = current_world_ptr->game_turn; - - /* Set correct dun_level value */ - sf_ptr->dun_level = creature_ptr->current_floor_ptr->dun_level; - - /* Create connected stairs */ - if (!(creature_ptr->change_floor_mode & CFM_NO_RETURN)) { - /* Extract stair position */ - grid_type *g_ptr = &creature_ptr->current_floor_ptr->grid_array[creature_ptr->y][creature_ptr->x]; - - /*** Create connected stairs ***/ - - /* No stairs down from Quest */ - if ((creature_ptr->change_floor_mode & CFM_UP) && !quest_number(creature_ptr, creature_ptr->current_floor_ptr->dun_level)) { - g_ptr->feat = (creature_ptr->change_floor_mode & CFM_SHAFT) ? feat_state(creature_ptr, feat_down_stair, FF_SHAFT) : feat_down_stair; - } - - /* No stairs up when ironman_downward */ - else if ((creature_ptr->change_floor_mode & CFM_DOWN) && !ironman_downward) { - g_ptr->feat = (creature_ptr->change_floor_mode & CFM_SHAFT) ? feat_state(creature_ptr, feat_up_stair, FF_SHAFT) : feat_up_stair; - } - - /* Paranoia -- Clear mimic */ - g_ptr->mimic = 0; - - /* Connect to previous floor */ - g_ptr->special = creature_ptr->floor_id; - } - } - - /* Arrive at random grid */ - if (creature_ptr->change_floor_mode & (CFM_RAND_PLACE)) { - (void)new_player_spot(creature_ptr); - } - - /* You see stairs blocked */ - else if ((creature_ptr->change_floor_mode & CFM_NO_RETURN) && (creature_ptr->change_floor_mode & (CFM_DOWN | CFM_UP))) { - if (!creature_ptr->blind) { - msg_print(_("突然階段が塞がれてしまった。", "Suddenly the stairs is blocked!")); - } else { - msg_print(_("ゴトゴトと何か音がした。", "You hear some noises.")); - } - } - - /* - * Update visit mark - * - * The "turn" is not always different number because - * the level teleport doesn't take any turn. Use - * visit mark instead of last visit turn to find the - * oldest saved floor. - */ - sf_ptr->visit_mark = latest_visit_mark++; - } - - /* Place preserved pet monsters */ - place_pet(creature_ptr); - - /* Reset travel target place */ - forget_travel_flow(creature_ptr->current_floor_ptr); - - /* Hack -- maintain unique and artifacts */ - update_unique_artifact(creature_ptr->current_floor_ptr, new_floor_id); - - /* Now the player is in new floor */ - creature_ptr->floor_id = new_floor_id; - - /* The dungeon is ready */ - current_world_ptr->character_dungeon = TRUE; - - /* Hack -- Munchkin characters always get whole map */ - if (creature_ptr->pseikaku == PERSONALITY_MUNCHKIN) - wiz_lite(creature_ptr, (bool)(creature_ptr->pclass == CLASS_NINJA)); - - /* Remember when this level was "created" */ - creature_ptr->current_floor_ptr->generated_turn = current_world_ptr->game_turn; - - /* No dungeon feeling yet */ - creature_ptr->feeling_turn = creature_ptr->current_floor_ptr->generated_turn; - creature_ptr->feeling = 0; - - /* Clear all flags */ - creature_ptr->change_floor_mode = 0L; - - select_floor_music(creature_ptr); - creature_ptr->change_floor_mode = 0; -}