OSDN Git Service

[Refactor] #39321 do_cmd_write_nikki() を exe_write_diary() に改名.
[hengband/hengband.git] / src / monster1.c
index 1c84f78..0f46a4d 100644 (file)
  */
 
 #include "angband.h"
+#include "util.h"
+#include "term.h"
+
+#include "cmd-dump.h"
+#include "bldg.h"
 #include "cmd-pet.h"
+#include "floor.h"
+#include "objectkind-hook.h"
+#include "player-personality.h"
+#include "monster.h"
+#include "monster-spell.h"
 #include "monsterrace-hook.h"
+#include "spells-summon.h"
+#include "patron.h"
+#include "quest.h"
+#include "artifact.h"
+#include "avatar.h"
+#include "wild.h"
+#include "spells.h"
+#include "dungeon.h"
+#include "world.h"
+#include "melee.h"
+#include "japanese.h"
+#include "view-mainwindow.h"
+#include "player-class.h"
+#include "english.h"
 
 
 /*
  * Pronoun arrays, by gender.
  */
-static cptr wd_he[3] =
+static concptr wd_he[3] =
 #ifdef JP
 { "それ", "彼", "彼女" };
 #else
 { "it", "he", "she" };
 #endif
 
-static cptr wd_his[3] =
+static concptr wd_his[3] =
 #ifdef JP
 { "それの", "彼の", "彼女の" };
 #else
@@ -58,17 +82,9 @@ static bool know_armour(MONRACE_IDX r_idx)
     bool known = (r_ptr->r_cast_spell == MAX_UCHAR)? TRUE: FALSE;
 
        if (cheat_know || known) return (TRUE);
-
-       /* Normal monsters */
        if (kills > 304 / (4 + level)) return (TRUE);
-
-       /* Skip non-uniques */
        if (!(r_ptr->flags1 & RF1_UNIQUE)) return (FALSE);
-
-       /* Unique monsters */
        if (kills > 304 / (38 + (5 * level) / 4)) return (TRUE);
-
-       /* Assume false */
        return (FALSE);
 }
 
@@ -99,14 +115,8 @@ static bool know_damage(MONRACE_IDX r_idx, int i)
        s32b d = d1 * d2;
 
        if (d >= ((4+level)*MAX_UCHAR)/80) d = ((4+level)*MAX_UCHAR-1)/80;
-
-       /* Normal monsters */
        if ((4 + level) * a > 80 * d) return (TRUE);
-
-       /* Skip non-uniques */
        if (!(r_ptr->flags1 & RF1_UNIQUE)) return (FALSE);
-
-       /* Unique monsters */
        if ((4 + level) * (2 * a) > 80 * d) return (TRUE);
 
        /* Assume false */
@@ -117,14 +127,14 @@ static bool know_damage(MONRACE_IDX r_idx, int i)
 /*
  * Prepare hook for c_roff(). It will be changed for spoiler generation in wizard1.c.
  */
-void (*hook_c_roff)(byte attr, cptr str) = c_roff;
+void (*hook_c_roff)(TERM_COLOR attr, concptr str) = c_roff;
 
 /*!
  * @brief モンスターの思い出メッセージをあらかじめ指定された関数ポインタに基づき出力する
  * @param str 出力文字列
  * @return なし
  */
-static void hooked_roff(cptr str)
+static void hooked_roff(concptr str)
 {
        /* Spawn */
        hook_c_roff(TERM_WHITE, str);
@@ -210,7 +220,7 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
        monster_race    *r_ptr = &r_info[r_idx];
        bool            old = FALSE;
        int             m, n, r;
-       cptr            p, q;
+       concptr            p, q;
 
 #ifdef JP
        char            jverb_buf[64];
@@ -239,7 +249,7 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
 
        int             vn = 0;
        byte            color[96];
-       cptr            vp[96];
+       concptr         vp[96];
        char tmp_msg[96][96];
 
        bool know_everything = FALSE;
@@ -258,7 +268,7 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
        flags7 = (r_ptr->flags7 & r_ptr->flags7);
        flagsr = (r_ptr->flagsr & r_ptr->r_flagsr);
 
-       for(n = 0; n < 6; n++)
+       for(n = 0; n < A_MAX; n++)
        {
                if(r_ptr->reinforce_id[n] > 0) reinforce = TRUE;
        }
@@ -418,8 +428,6 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
                /* Start a new line */
                hooked_roff("\n");
        }
-
-       /* Normal monsters */
        else
        {
                /* Killed some this life */
@@ -450,7 +458,7 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
 
        /* Descriptions */
        {
-               cptr tmp = r_text + r_ptr->text;
+               concptr tmp = r_text + r_ptr->text;
 
                if (tmp[0])
                {
@@ -543,7 +551,6 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
                        if (speed != 110) hooked_roff(_("、かつ", ", and"));
                }
 
-               /* Speed */
                if (speed > 110)
                {
                        if (speed > 139) hook_c_roff(TERM_RED, _("信じ難いほど", " incredibly"));
@@ -738,7 +745,7 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
 #ifndef JP
                        hooked_roff(" contain ");
 #endif                 
-                       for(n = 0; n < 6; n++)
+                       for(n = 0; n < A_MAX; n++)
                        {
                                if(r_ptr->reinforce_id[n] && r_ptr->reinforce_dd[n] && r_ptr->reinforce_ds[n])
                                {
@@ -750,11 +757,10 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
                                        else
                                        {
 #ifdef JP
-                                               hooked_roff(format("、 %dd%d 体の%s", r_ptr->reinforce_dd[n], r_ptr->reinforce_ds[n],
-                                                       r_name + rf_ptr->name));
+                                               hooked_roff(format("、 %dd%d 体の%s", r_ptr->reinforce_dd[n], r_ptr->reinforce_ds[n], r_name + rf_ptr->name));
 #else
                                                bool plural = (r_ptr->reinforce_dd[n] * r_ptr->reinforce_ds[n] > 1);
-                                               char name[80];
+                                               GAME_TEXT name[MAX_NLEN];
                                                strcpy(name, r_name + rf_ptr->name);
                                                if(plural) plural_aux(name);
                                                hooked_roff(format(",%dd%d %s", r_ptr->reinforce_dd[n], r_ptr->reinforce_ds[n], name));
@@ -1591,7 +1597,7 @@ static void roff_aux(MONRACE_IDX r_idx, BIT_FLAGS mode)
                  (r_ptr->r_ignore == MAX_UCHAR) ||
            (r_ptr->sleep == 0 && r_ptr->r_tkills >= 10) || know_everything)
        {
-               cptr act;
+               concptr act;
 
                if (r_ptr->sleep > 200)
                {
@@ -2058,8 +2064,7 @@ void roff_top(MONRACE_IDX r_idx)
  */
 void screen_roff(MONRACE_IDX r_idx, BIT_FLAGS mode)
 {
-       /* Flush messages */
-       msg_print(NULL);
+       msg_erase();
 
        /* Begin recall */
        Term_erase(0, 1, 255);
@@ -2113,7 +2118,7 @@ void display_roff(MONRACE_IDX r_idx)
  * @param roff_func 出力処理を行う関数ポインタ
  * @return なし
  */
-void output_monster_spoiler(MONRACE_IDX r_idx, void (*roff_func)(byte attr, cptr str))
+void output_monster_spoiler(MONRACE_IDX r_idx, void (*roff_func)(TERM_COLOR attr, concptr str))
 {
        hook_c_roff = roff_func;
 
@@ -2129,7 +2134,7 @@ void output_monster_spoiler(MONRACE_IDX r_idx, void (*roff_func)(byte attr, cptr
  */
 monsterrace_hook_type get_monster_hook(void)
 {
-       if (!dun_level && !p_ptr->inside_quest)
+       if (!current_floor_ptr->dun_level && !p_ptr->inside_quest)
        {
                switch (wilderness[p_ptr->wilderness_y][p_ptr->wilderness_x].terrain)
                {
@@ -2168,7 +2173,7 @@ monsterrace_hook_type get_monster_hook(void)
  */
 monsterrace_hook_type get_monster_hook2(POSITION y, POSITION x)
 {
-       feature_type *f_ptr = &f_info[cave[y][x].feat];
+       feature_type *f_ptr = &f_info[current_floor_ptr->grid_array[y][x].feat];
 
        /* Set the monster list */
 
@@ -2214,9 +2219,6 @@ void set_friendly(monster_type *m_ptr)
  */
 void set_pet(monster_type *m_ptr)
 {
-       if (!is_pet(m_ptr)) check_pets_num_and_align(m_ptr, TRUE);
-
-       /* Check for quest completion */
        check_quest_completion(m_ptr);
 
        m_ptr->smart |= SM_PET;
@@ -2232,10 +2234,7 @@ void set_pet(monster_type *m_ptr)
  */
 void set_hostile(monster_type *m_ptr)
 {
-       if (p_ptr->inside_battle) return;
-
-       if (is_pet(m_ptr)) check_pets_num_and_align(m_ptr, FALSE);
-
+       if (p_ptr->phase_out) return;
        m_ptr->smart &= ~SM_PET;
        m_ptr->smart &= ~SM_FRIENDLY;
 }
@@ -2249,20 +2248,20 @@ void set_hostile(monster_type *m_ptr)
  */
 void anger_monster(monster_type *m_ptr)
 {
-       if (p_ptr->inside_battle) return;
+       if (p_ptr->phase_out) return;
        if (is_friendly(m_ptr))
        {
-               char m_name[80];
+               GAME_TEXT m_name[MAX_NLEN];
 
                monster_desc(m_name, m_ptr, 0);
                msg_format(_("%^sは怒った!", "%^s gets angry!"), m_name);
 
                set_hostile(m_ptr);
 
-               chg_virtue(V_INDIVIDUALISM, 1);
-               chg_virtue(V_HONOUR, -1);
-               chg_virtue(V_JUSTICE, -1);
-               chg_virtue(V_COMPASSION, -1);
+               chg_virtue(p_ptr, V_INDIVIDUALISM, 1);
+               chg_virtue(p_ptr, V_HONOUR, -1);
+               chg_virtue(p_ptr, V_JUSTICE, -1);
+               chg_virtue(p_ptr, V_COMPASSION, -1);
        }
 }
 
@@ -2279,7 +2278,6 @@ bool monster_can_cross_terrain(FEAT_IDX feat, monster_race *r_ptr, BIT_FLAGS16 m
 {
        feature_type *f_ptr = &f_info[feat];
 
-       /* Pattern */
        if (have_flag(f_ptr->flags, FF_PATTERN))
        {
                if (!(mode & CEM_RIDING))
@@ -2327,6 +2325,30 @@ bool monster_can_cross_terrain(FEAT_IDX feat, monster_race *r_ptr, BIT_FLAGS16 m
                if (!(r_ptr->flagsr & RFR_EFF_IM_FIRE_MASK)) return FALSE;
        }
 
+       /* Cold */
+       if (have_flag(f_ptr->flags, FF_COLD_PUDDLE))
+       {
+               if (!(r_ptr->flagsr & RFR_EFF_IM_COLD_MASK)) return FALSE;
+       }
+
+       /* Elec */
+       if (have_flag(f_ptr->flags, FF_ELEC_PUDDLE))
+       {
+               if (!(r_ptr->flagsr & RFR_EFF_IM_ELEC_MASK)) return FALSE;
+       }
+
+       /* Acid */
+       if (have_flag(f_ptr->flags, FF_ACID_PUDDLE))
+       {
+               if (!(r_ptr->flagsr & RFR_EFF_IM_ACID_MASK)) return FALSE;
+       }
+
+       /* Poison */
+       if (have_flag(f_ptr->flags, FF_POISON_PUDDLE))
+       {
+               if (!(r_ptr->flagsr & RFR_EFF_IM_POIS_MASK)) return FALSE;
+       }
+
        return TRUE;
 }
 
@@ -2342,13 +2364,13 @@ bool monster_can_cross_terrain(FEAT_IDX feat, monster_race *r_ptr, BIT_FLAGS16 m
  */
 bool monster_can_enter(POSITION y, POSITION x, monster_race *r_ptr, BIT_FLAGS16 mode)
 {
-       cave_type *c_ptr = &cave[y][x];
+       grid_type *g_ptr = &current_floor_ptr->grid_array[y][x];
 
        /* Player or other monster */
        if (player_bold(y, x)) return FALSE;
-       if (c_ptr->m_idx) return FALSE;
+       if (g_ptr->m_idx) return FALSE;
 
-       return monster_can_cross_terrain(c_ptr->feat, r_ptr, mode);
+       return monster_can_cross_terrain(g_ptr->feat, r_ptr, mode);
 }
 
 
@@ -2385,7 +2407,7 @@ bool are_enemies(monster_type *m_ptr, monster_type *n_ptr)
        monster_race *r_ptr = &r_info[m_ptr->r_idx];
        monster_race *s_ptr = &r_info[n_ptr->r_idx];
 
-       if (p_ptr->inside_battle)
+       if (p_ptr->phase_out)
        {
                if (is_pet(m_ptr) || is_pet(n_ptr)) return FALSE;
                return TRUE;
@@ -2449,3 +2471,756 @@ bool monster_has_hostile_align(monster_type *m_ptr, int pa_good, int pa_evil, mo
        /* Non-hostile alignment */
        return FALSE;
 }
+
+/*!
+ * @brief モンスターを倒した際の財宝svalを返す
+ * @param r_idx 倒したモンスターの種族ID
+ * @return 財宝のsval
+ * @details
+ * Hack -- Return the "automatic coin type" of a monster race
+ * Used to allocate proper treasure when "Creeping coins" die
+ * Note the use of actual "monster names"
+ */
+static OBJECT_SUBTYPE_VALUE get_coin_type(MONRACE_IDX r_idx)
+{
+       /* Analyze monsters */
+       switch (r_idx)
+       {
+       case MON_COPPER_COINS: return 2;
+       case MON_SILVER_COINS: return 5;
+       case MON_GOLD_COINS: return 10;
+       case MON_MITHRIL_COINS:
+       case MON_MITHRIL_GOLEM: return 16;
+       case MON_ADAMANT_COINS: return 17;
+       }
+
+       /* Assume nothing */
+       return 0;
+}
+
+
+/*!
+ * @brief モンスターが死亡した時の処理 /
+ * Handle the "death" of a monster.
+ * @param m_idx 死亡したモンスターのID
+ * @param drop_item TRUEならばモンスターのドロップ処理を行う
+ * @return 撃破されたモンスターの述語
+ * @details
+ * <pre>
+ * Disperse treasures centered at the monster location based on the
+ * various flags contained in the monster flags fields.
+ * Check for "Quest" completion when a quest monster is killed.
+ * Note that only the player can induce "monster_death()" on Uniques.
+ * Thus (for now) all Quest monsters should be Uniques.
+ * Note that monsters can now carry objects, and when a monster dies,
+ * it drops all of its objects, which may disappear in crowded rooms.
+ * </pre>
+ */
+void monster_death(MONSTER_IDX m_idx, bool drop_item)
+{
+       int i, j;
+       POSITION y, x;
+
+       int dump_item = 0;
+       int dump_gold = 0;
+       int number = 0;
+
+       monster_type *m_ptr = &current_floor_ptr->m_list[m_idx];
+       monster_race *r_ptr = &r_info[m_ptr->r_idx];
+
+       bool visible = ((m_ptr->ml && !p_ptr->image) || (r_ptr->flags1 & RF1_UNIQUE));
+
+       u32b mo_mode = 0L;
+
+       bool do_gold = (!(r_ptr->flags1 & RF1_ONLY_ITEM));
+       bool do_item = (!(r_ptr->flags1 & RF1_ONLY_GOLD));
+       bool cloned = (m_ptr->smart & SM_CLONED) ? TRUE : FALSE;
+       int force_coin = get_coin_type(m_ptr->r_idx);
+
+       object_type forge;
+       object_type *q_ptr;
+
+       bool drop_chosen_item = drop_item && !cloned && !p_ptr->inside_arena
+               && !p_ptr->phase_out && !is_pet(m_ptr);
+
+       /* The caster is dead? */
+       if (current_world_ptr->timewalk_m_idx && current_world_ptr->timewalk_m_idx == m_idx) current_world_ptr->timewalk_m_idx = 0;
+
+       /* Notice changes in view */
+       if (r_ptr->flags7 & (RF7_LITE_MASK | RF7_DARK_MASK))
+       {
+               p_ptr->update |= (PU_MON_LITE);
+       }
+
+       y = m_ptr->fy;
+       x = m_ptr->fx;
+
+       if (record_named_pet && is_pet(m_ptr) && m_ptr->nickname)
+       {
+               GAME_TEXT m_name[MAX_NLEN];
+
+               monster_desc(m_name, m_ptr, MD_INDEF_VISIBLE);
+               exe_write_diary(p_ptr, NIKKI_NAMED_PET, 3, m_name);
+       }
+
+       /* Let monsters explode! */
+       for (i = 0; i < 4; i++)
+       {
+               if (r_ptr->blow[i].method == RBM_EXPLODE)
+               {
+                       BIT_FLAGS flg = PROJECT_GRID | PROJECT_ITEM | PROJECT_KILL;
+                       EFFECT_ID typ = mbe_info[r_ptr->blow[i].effect].explode_type;
+                       DICE_NUMBER d_dice = r_ptr->blow[i].d_dice;
+                       DICE_SID d_side = r_ptr->blow[i].d_side;
+                       HIT_POINT damage = damroll(d_dice, d_side);
+
+                       project(m_idx, 3, y, x, damage, typ, flg, -1);
+                       break;
+               }
+       }
+
+       if (m_ptr->mflag2 & MFLAG2_CHAMELEON)
+       {
+               choose_new_monster(m_idx, TRUE, MON_CHAMELEON);
+               r_ptr = &r_info[m_ptr->r_idx];
+       }
+
+       check_quest_completion(m_ptr);
+
+       /* Handle the possibility of player vanquishing arena combatant -KMW- */
+       if (p_ptr->inside_arena && !is_pet(m_ptr))
+       {
+               p_ptr->exit_bldg = TRUE;
+
+               if (p_ptr->arena_number > MAX_ARENA_MONS)
+               {
+                       msg_print(_("素晴らしい!君こそ真の勝利者だ。", "You are a Genuine Champion!"));
+               }
+               else
+               {
+                       msg_print(_("勝利!チャンピオンへの道を進んでいる。", "Victorious! You're on your way to becoming Champion."));
+               }
+
+               if (arena_info[p_ptr->arena_number].tval)
+               {
+                       q_ptr = &forge;
+
+                       /* Prepare to make a prize */
+                       object_prep(q_ptr, lookup_kind(arena_info[p_ptr->arena_number].tval, arena_info[p_ptr->arena_number].sval));
+                       apply_magic(q_ptr, current_floor_ptr->object_level, AM_NO_FIXED_ART);
+                       (void)drop_near(q_ptr, -1, y, x);
+               }
+
+               if (p_ptr->arena_number > MAX_ARENA_MONS) p_ptr->arena_number++;
+               p_ptr->arena_number++;
+               if (record_arena)
+               {
+                       GAME_TEXT m_name[MAX_NLEN];
+
+                       monster_desc(m_name, m_ptr, MD_WRONGDOER_NAME);
+
+                       exe_write_diary(p_ptr, NIKKI_ARENA, p_ptr->arena_number, m_name);
+               }
+       }
+
+       if (m_idx == p_ptr->riding)
+       {
+               if (rakuba(-1, FALSE))
+               {
+                       msg_print(_("地面に落とされた。", "You have fallen from your riding pet."));
+               }
+       }
+
+       /* Drop a dead corpse? */
+       if (one_in_(r_ptr->flags1 & RF1_UNIQUE ? 1 : 4) &&
+               (r_ptr->flags9 & (RF9_DROP_CORPSE | RF9_DROP_SKELETON)) &&
+               !(p_ptr->inside_arena || p_ptr->phase_out || cloned || ((m_ptr->r_idx == today_mon) && is_pet(m_ptr))))
+       {
+               /* Assume skeleton */
+               bool corpse = FALSE;
+
+               /*
+                * We cannot drop a skeleton? Note, if we are in this check,
+                * we *know* we can drop at least a corpse or a skeleton
+                */
+               if (!(r_ptr->flags9 & RF9_DROP_SKELETON))
+                       corpse = TRUE;
+               else if ((r_ptr->flags9 & RF9_DROP_CORPSE) && (r_ptr->flags1 & RF1_UNIQUE))
+                       corpse = TRUE;
+
+               /* Else, a corpse is more likely unless we did a "lot" of damage */
+               else if (r_ptr->flags9 & RF9_DROP_CORPSE)
+               {
+                       /* Lots of damage in one blow */
+                       if ((0 - ((m_ptr->maxhp) / 4)) > m_ptr->hp)
+                       {
+                               if (one_in_(5)) corpse = TRUE;
+                       }
+                       else
+                       {
+                               if (!one_in_(5)) corpse = TRUE;
+                       }
+               }
+               q_ptr = &forge;
+
+               /* Prepare to make an object */
+               object_prep(q_ptr, lookup_kind(TV_CORPSE, (corpse ? SV_CORPSE : SV_SKELETON)));
+
+               apply_magic(q_ptr, current_floor_ptr->object_level, AM_NO_FIXED_ART);
+
+               q_ptr->pval = m_ptr->r_idx;
+               (void)drop_near(q_ptr, -1, y, x);
+       }
+
+       /* Drop objects being carried */
+       monster_drop_carried_objects(m_ptr);
+
+       if (r_ptr->flags1 & RF1_DROP_GOOD) mo_mode |= AM_GOOD;
+       if (r_ptr->flags1 & RF1_DROP_GREAT) mo_mode |= AM_GREAT;
+
+       switch (m_ptr->r_idx)
+       {
+       case MON_PINK_HORROR:
+               /* Pink horrors are replaced with 2 Blue horrors */
+               if (!(p_ptr->inside_arena || p_ptr->phase_out))
+               {
+                       bool notice = FALSE;
+
+                       for (i = 0; i < 2; i++)
+                       {
+                               POSITION wy = y, wx = x;
+                               bool pet = is_pet(m_ptr);
+                               BIT_FLAGS mode = 0L;
+
+                               if (pet) mode |= PM_FORCE_PET;
+
+                               if (summon_specific((pet ? -1 : m_idx), wy, wx, 100, SUMMON_BLUE_HORROR, mode))
+                               {
+                                       if (player_can_see_bold(wy, wx)) notice = TRUE;
+                               }
+                       }
+
+                       if (notice) msg_print(_("ピンク・ホラーは分裂した!", "The Pink horror divides!"));
+               }
+               break;
+
+       case MON_BLOODLETTER:
+               /* Bloodletters of Khorne may drop a blade of chaos */
+               if (drop_chosen_item && (randint1(100) < 15))
+               {
+                       q_ptr = &forge;
+
+                       /* Prepare to make a Blade of Chaos */
+                       object_prep(q_ptr, lookup_kind(TV_SWORD, SV_BLADE_OF_CHAOS));
+
+                       apply_magic(q_ptr, current_floor_ptr->object_level, AM_NO_FIXED_ART | mo_mode);
+                       (void)drop_near(q_ptr, -1, y, x);
+               }
+               break;
+
+       case MON_RAAL:
+               if (drop_chosen_item && (current_floor_ptr->dun_level > 9))
+               {
+                       q_ptr = &forge;
+                       object_wipe(q_ptr);
+
+                       /* Activate restriction */
+                       if ((current_floor_ptr->dun_level > 49) && one_in_(5))
+                               get_obj_num_hook = kind_is_good_book;
+                       else
+                               get_obj_num_hook = kind_is_book;
+
+                       /* Make a book */
+                       make_object(q_ptr, mo_mode);
+                       (void)drop_near(q_ptr, -1, y, x);
+               }
+               break;
+
+       case MON_DAWN:
+               /*
+                * Mega^3-hack: killing a 'Warrior of the Dawn' is likely to
+                * spawn another in the fallen one's place!
+                */
+               if (!p_ptr->inside_arena && !p_ptr->phase_out)
+               {
+                       if (!one_in_(7))
+                       {
+                               POSITION wy = y, wx = x;
+                               int attempts = 100;
+                               bool pet = is_pet(m_ptr);
+
+                               do
+                               {
+                                       scatter(&wy, &wx, y, x, 20, 0);
+                               } while (!(in_bounds(wy, wx) && cave_empty_bold2(wy, wx)) && --attempts);
+
+                               if (attempts > 0)
+                               {
+                                       BIT_FLAGS mode = 0L;
+                                       if (pet) mode |= PM_FORCE_PET;
+
+                                       if (summon_specific((pet ? -1 : m_idx), wy, wx, 100, SUMMON_DAWN, mode))
+                                       {
+                                               if (player_can_see_bold(wy, wx))
+                                                       msg_print(_("新たな戦士が現れた!", "A new warrior steps forth!"));
+                                       }
+                               }
+                       }
+               }
+               break;
+
+       case MON_UNMAKER:
+               /* One more ultra-hack: An Unmaker goes out with a big bang! */
+       {
+               BIT_FLAGS flg = PROJECT_GRID | PROJECT_ITEM | PROJECT_KILL;
+               (void)project(m_idx, 6, y, x, 100, GF_CHAOS, flg, -1);
+       }
+       break;
+
+       case MON_UNICORN_ORD:
+       case MON_MORGOTH:
+       case MON_ONE_RING:
+               /* Reward for "lazy" player */
+               if (p_ptr->pseikaku == SEIKAKU_NAMAKE)
+               {
+                       ARTIFACT_IDX a_idx = 0;
+                       artifact_type *a_ptr = NULL;
+
+                       if (!drop_chosen_item) break;
+
+                       do
+                       {
+                               switch (randint0(3))
+                               {
+                               case 0:
+                                       a_idx = ART_NAMAKE_HAMMER;
+                                       break;
+                               case 1:
+                                       a_idx = ART_NAMAKE_BOW;
+                                       break;
+                               case 2:
+                                       a_idx = ART_NAMAKE_ARMOR;
+                                       break;
+                               }
+
+                               a_ptr = &a_info[a_idx];
+                       } while (a_ptr->cur_num);
+
+                       if (create_named_art(a_idx, y, x))
+                       {
+                               a_ptr->cur_num = 1;
+
+                               /* Hack -- Memorize location of artifact in saved floors */
+                               if (current_world_ptr->character_dungeon) a_ptr->floor_id = p_ptr->floor_id;
+                       }
+                       else if (!preserve_mode) a_ptr->cur_num = 1;
+               }
+               break;
+
+       case MON_SERPENT:
+               if (!drop_chosen_item) break;
+               q_ptr = &forge;
+
+               /* Mega-Hack -- Prepare to make "Grond" */
+               object_prep(q_ptr, lookup_kind(TV_HAFTED, SV_GROND));
+
+               /* Mega-Hack -- Mark this item as "Grond" */
+               q_ptr->name1 = ART_GROND;
+
+               /* Mega-Hack -- Actually create "Grond" */
+               apply_magic(q_ptr, -1, AM_GOOD | AM_GREAT);
+               (void)drop_near(q_ptr, -1, y, x);
+               q_ptr = &forge;
+
+               /* Mega-Hack -- Prepare to make "Chaos" */
+               object_prep(q_ptr, lookup_kind(TV_CROWN, SV_CHAOS));
+
+               /* Mega-Hack -- Mark this item as "Chaos" */
+               q_ptr->name1 = ART_CHAOS;
+
+               /* Mega-Hack -- Actually create "Chaos" */
+               apply_magic(q_ptr, -1, AM_GOOD | AM_GREAT);
+               (void)drop_near(q_ptr, -1, y, x);
+               break;
+
+       case MON_B_DEATH_SWORD:
+               if (drop_chosen_item)
+               {
+                       q_ptr = &forge;
+
+                       /* Prepare to make a broken sword */
+                       object_prep(q_ptr, lookup_kind(TV_SWORD, randint1(2)));
+                       (void)drop_near(q_ptr, -1, y, x);
+               }
+               break;
+
+       case MON_A_GOLD:
+       case MON_A_SILVER:
+               if (drop_chosen_item && ((m_ptr->r_idx == MON_A_GOLD) ||
+                       ((m_ptr->r_idx == MON_A_SILVER) && (r_ptr->r_akills % 5 == 0))))
+               {
+                       q_ptr = &forge;
+
+                       /* Prepare to make a Can of Toys */
+                       object_prep(q_ptr, lookup_kind(TV_CHEST, SV_CHEST_KANDUME));
+
+                       apply_magic(q_ptr, current_floor_ptr->object_level, AM_NO_FIXED_ART);
+                       (void)drop_near(q_ptr, -1, y, x);
+               }
+               break;
+
+       case MON_ROLENTO:
+       {
+               BIT_FLAGS flg = PROJECT_GRID | PROJECT_ITEM | PROJECT_KILL;
+               (void)project(m_idx, 3, y, x, damroll(20, 10), GF_FIRE, flg, -1);
+       }
+       break;
+
+       default:
+               if (!drop_chosen_item) break;
+
+               switch (r_ptr->d_char)
+               {
+               case '(':
+                       if (current_floor_ptr->dun_level > 0)
+                       {
+                               q_ptr = &forge;
+                               object_wipe(q_ptr);
+
+                               /* Activate restriction */
+                               get_obj_num_hook = kind_is_cloak;
+
+                               /* Make a cloak */
+                               make_object(q_ptr, mo_mode);
+                               (void)drop_near(q_ptr, -1, y, x);
+                       }
+                       break;
+
+               case '/':
+                       if (current_floor_ptr->dun_level > 4)
+                       {
+                               q_ptr = &forge;
+                               object_wipe(q_ptr);
+
+                               /* Activate restriction */
+                               get_obj_num_hook = kind_is_polearm;
+
+                               /* Make a poleweapon */
+                               make_object(q_ptr, mo_mode);
+                               (void)drop_near(q_ptr, -1, y, x);
+                       }
+                       break;
+
+               case '[':
+                       if (current_floor_ptr->dun_level > 19)
+                       {
+                               q_ptr = &forge;
+                               object_wipe(q_ptr);
+
+                               /* Activate restriction */
+                               get_obj_num_hook = kind_is_armor;
+
+                               /* Make a hard armor */
+                               make_object(q_ptr, mo_mode);
+                               (void)drop_near(q_ptr, -1, y, x);
+                       }
+                       break;
+
+               case '\\':
+                       if (current_floor_ptr->dun_level > 4)
+                       {
+                               q_ptr = &forge;
+                               object_wipe(q_ptr);
+
+                               /* Activate restriction */
+                               get_obj_num_hook = kind_is_hafted;
+
+                               /* Make a hafted weapon */
+                               make_object(q_ptr, mo_mode);
+                               (void)drop_near(q_ptr, -1, y, x);
+                       }
+                       break;
+
+               case '|':
+                       if (m_ptr->r_idx != MON_STORMBRINGER)
+                       {
+                               q_ptr = &forge;
+                               object_wipe(q_ptr);
+
+                               /* Activate restriction */
+                               get_obj_num_hook = kind_is_sword;
+
+                               /* Make a sword */
+                               make_object(q_ptr, mo_mode);
+                               (void)drop_near(q_ptr, -1, y, x);
+                       }
+                       break;
+               }
+               break;
+       }
+
+       /* Mega-Hack -- drop fixed items */
+       if (drop_chosen_item)
+       {
+               ARTIFACT_IDX a_idx = 0;
+               PERCENTAGE chance = 0;
+
+               for (i = 0; i < 4; i++)
+               {
+                       if (!r_ptr->artifact_id[i]) break;
+                       a_idx = r_ptr->artifact_id[i];
+                       chance = r_ptr->artifact_percent[i];
+               }
+
+               if ((a_idx > 0) && ((randint0(100) < chance) || p_ptr->wizard))
+               {
+                       artifact_type *a_ptr = &a_info[a_idx];
+
+                       if (!a_ptr->cur_num)
+                       {
+                               if (create_named_art(a_idx, y, x))
+                               {
+                                       a_ptr->cur_num = 1;
+
+                                       /* Hack -- Memorize location of artifact in saved floors */
+                                       if (current_world_ptr->character_dungeon) a_ptr->floor_id = p_ptr->floor_id;
+                               }
+                               else if (!preserve_mode) a_ptr->cur_num = 1;
+                       }
+               }
+
+               if ((r_ptr->flags7 & RF7_GUARDIAN) && (d_info[p_ptr->dungeon_idx].final_guardian == m_ptr->r_idx))
+               {
+                       KIND_OBJECT_IDX k_idx = d_info[p_ptr->dungeon_idx].final_object ? d_info[p_ptr->dungeon_idx].final_object
+                               : lookup_kind(TV_SCROLL, SV_SCROLL_ACQUIREMENT);
+
+                       if (d_info[p_ptr->dungeon_idx].final_artifact)
+                       {
+                               a_idx = d_info[p_ptr->dungeon_idx].final_artifact;
+                               artifact_type *a_ptr = &a_info[a_idx];
+
+                               if (!a_ptr->cur_num)
+                               {
+                                       if (create_named_art(a_idx, y, x))
+                                       {
+                                               a_ptr->cur_num = 1;
+
+                                               /* Hack -- Memorize location of artifact in saved floors */
+                                               if (current_world_ptr->character_dungeon) a_ptr->floor_id = p_ptr->floor_id;
+                                       }
+                                       else if (!preserve_mode) a_ptr->cur_num = 1;
+
+                                       /* Prevent rewarding both artifact and "default" object */
+                                       if (!d_info[p_ptr->dungeon_idx].final_object) k_idx = 0;
+                               }
+                       }
+
+                       if (k_idx)
+                       {
+                               q_ptr = &forge;
+
+                               /* Prepare to make a reward */
+                               object_prep(q_ptr, k_idx);
+
+                               apply_magic(q_ptr, current_floor_ptr->object_level, AM_NO_FIXED_ART | AM_GOOD);
+                               (void)drop_near(q_ptr, -1, y, x);
+                       }
+                       msg_format(_("あなたは%sを制覇した!", "You have conquered %s!"), d_name + d_info[p_ptr->dungeon_idx].name);
+               }
+       }
+
+       /* Determine how much we can drop */
+       if ((r_ptr->flags1 & RF1_DROP_60) && (randint0(100) < 60)) number++;
+       if ((r_ptr->flags1 & RF1_DROP_90) && (randint0(100) < 90)) number++;
+       if (r_ptr->flags1 & RF1_DROP_1D2) number += damroll(1, 2);
+       if (r_ptr->flags1 & RF1_DROP_2D2) number += damroll(2, 2);
+       if (r_ptr->flags1 & RF1_DROP_3D2) number += damroll(3, 2);
+       if (r_ptr->flags1 & RF1_DROP_4D2) number += damroll(4, 2);
+
+       if (cloned && !(r_ptr->flags1 & RF1_UNIQUE))
+               number = 0; /* Clones drop no stuff unless Cloning Pits */
+
+       if (is_pet(m_ptr) || p_ptr->phase_out || p_ptr->inside_arena)
+               number = 0; /* Pets drop no stuff */
+       if (!drop_item && (r_ptr->d_char != '$')) number = 0;
+
+       if ((r_ptr->flags2 & (RF2_MULTIPLY)) && (r_ptr->r_akills > 1024))
+               number = 0; /* Limit of Multiply monster drop */
+
+       /* Hack -- handle creeping coins */
+       coin_type = force_coin;
+
+       /* Average dungeon and monster levels */
+       current_floor_ptr->object_level = (current_floor_ptr->dun_level + r_ptr->level) / 2;
+
+       /* Drop some objects */
+       for (j = 0; j < number; j++)
+       {
+               q_ptr = &forge;
+               object_wipe(q_ptr);
+
+               if (do_gold && (!do_item || (randint0(100) < 50)))
+               {
+                       if (!make_gold(q_ptr)) continue;
+                       dump_gold++;
+               }
+               else
+               {
+                       if (!make_object(q_ptr, mo_mode)) continue;
+                       dump_item++;
+               }
+               (void)drop_near(q_ptr, -1, y, x);
+       }
+
+       /* Reset the object level */
+       current_floor_ptr->object_level = current_floor_ptr->base_level;
+
+       /* Reset "coin" type */
+       coin_type = 0;
+
+
+       /* Take note of any dropped treasure */
+       if (visible && (dump_item || dump_gold))
+       {
+               /* Take notes on treasure */
+               lore_treasure(m_idx, dump_item, dump_gold);
+       }
+
+       /* Only process "Quest Monsters" */
+       if (!(r_ptr->flags1 & RF1_QUESTOR)) return;
+       if (p_ptr->phase_out) return;
+
+       /* Winner? */
+       if ((m_ptr->r_idx == MON_SERPENT) && !cloned)
+       {
+               /* Total winner */
+               p_ptr->total_winner = TRUE;
+
+               /* Redraw the "title" */
+               p_ptr->redraw |= (PR_TITLE);
+
+               play_music(TERM_XTRA_MUSIC_BASIC, MUSIC_BASIC_FINAL_QUEST_CLEAR);
+
+               exe_write_diary(p_ptr, NIKKI_BUNSHOU, 0, _("見事に変愚蛮怒の勝利者となった!", "become *WINNER* of Hengband finely!"));
+
+               admire_from_patron(p_ptr);
+
+               /* Congratulations */
+               msg_print(_("*** おめでとう ***", "*** CONGRATULATIONS ***"));
+               msg_print(_("あなたはゲームをコンプリートしました。", "You have won the game!"));
+               msg_print(_("準備が整ったら引退(自殺コマンド)しても結構です。", "You may retire (commit suicide) when you are ready."));
+       }
+}
+
+/*!
+ * @brief モンスターを撃破した際の述語メッセージを返す /
+ * Return monster death string
+ * @param r_ptr 撃破されたモンスターの種族情報を持つ構造体の参照ポインタ
+ * @return 撃破されたモンスターの述語
+ */
+concptr extract_note_dies(MONRACE_IDX r_idx)
+{
+       monster_race *r_ptr = &r_info[r_idx];
+       /* Some monsters get "destroyed" */
+       if (!monster_living(r_idx))
+       {
+               int i;
+
+               for (i = 0; i < 4; i++)
+               {
+                       if (r_ptr->blow[i].method == RBM_EXPLODE)
+                       {
+                               return _("は爆発して粉々になった。", " explodes into tiny shreds.");
+                       }
+               }
+               return _("を倒した。", " is destroyed.");
+       }
+
+       return _("は死んだ。", " dies.");
+}
+
+/*
+ * Monster health description
+ */
+concptr look_mon_desc(monster_type *m_ptr, BIT_FLAGS mode)
+{
+       monster_race *ap_r_ptr = &r_info[m_ptr->ap_r_idx];
+       bool living;
+       int perc;
+       concptr desc;
+       concptr attitude;
+       concptr clone;
+
+       /* Determine if the monster is "living" */
+       living = monster_living(m_ptr->ap_r_idx);
+
+       /* Calculate a health "percentage" */
+       perc = m_ptr->maxhp > 0 ? 100L * m_ptr->hp / m_ptr->maxhp : 0;
+
+       /* Healthy monsters */
+       if (m_ptr->hp >= m_ptr->maxhp)
+       {
+               desc = living ? _("無傷", "unhurt") : _("無ダメージ", "undamaged");
+       }
+
+       else if (perc >= 60)
+       {
+               desc = living ? _("軽傷", "somewhat wounded") : _("小ダメージ", "somewhat damaged");
+       }
+
+       else if (perc >= 25)
+       {
+               desc = living ? _("負傷", "wounded") : _("中ダメージ", "damaged");
+       }
+
+       else if (perc >= 10)
+       {
+               desc = living ? _("重傷", "badly wounded") : _("大ダメージ", "badly damaged");
+       }
+
+       else
+       {
+               desc = living ? _("半死半生", "almost dead") : _("倒れかけ", "almost destroyed");
+       }
+
+       /* Need attitude information? */
+       if (!(mode & 0x01))
+       {
+               /* Full information is not needed */
+               attitude = "";
+       }
+       else if (is_pet(m_ptr))
+       {
+               attitude = _(", ペット", ", pet");
+       }
+       else if (is_friendly(m_ptr))
+       {
+               attitude = _(", 友好的", ", friendly");
+       }
+       else
+       {
+               attitude = _("", "");
+       }
+
+       /* Clone monster? */
+       if (m_ptr->smart & SM_CLONED)
+       {
+               clone = ", clone";
+       }
+       else
+       {
+               clone = "";
+       }
+
+       /* Display monster's level --- idea borrowed from ToME */
+       if (ap_r_ptr->r_tkills && !(m_ptr->mflag2 & MFLAG2_KAGE))
+       {
+               return format(_("レベル%d, %s%s%s", "Level %d, %s%s%s"), ap_r_ptr->level, desc, attitude, clone);
+       }
+       else
+       {
+               return format(_("レベル???, %s%s%s", "Level ???, %s%s%s"), desc, attitude, clone);
+       }
+
+}
+