4 #include "player-status.h"
5 #include "object-hook.h"
6 #include "monster-status.h"
9 * @brief プレイヤーの騎乗/下馬処理判定
10 * @param g_ptr プレイヤーの移動先マスの構造体参照ポインタ
11 * @param now_riding TRUEなら下馬処理、FALSEならば騎乗処理
12 * @return 可能ならばTRUEを返す
14 bool player_can_ride_aux(grid_type *g_ptr, bool now_riding)
17 bool old_character_xtra = character_xtra;
18 MONSTER_IDX old_riding = p_ptr->riding;
19 bool old_riding_ryoute = p_ptr->riding_ryoute;
20 bool old_old_riding_ryoute = p_ptr->old_riding_ryoute;
21 bool old_pf_ryoute = (p_ptr->pet_extra_flags & PF_RYOUTE) ? TRUE : FALSE;
23 /* Hack -- prevent "icky" message */
24 character_xtra = TRUE;
26 if (now_riding) p_ptr->riding = g_ptr->m_idx;
30 p_ptr->pet_extra_flags &= ~(PF_RYOUTE);
31 p_ptr->riding_ryoute = p_ptr->old_riding_ryoute = FALSE;
34 p_ptr->update |= PU_BONUS;
37 p_can_enter = player_can_enter(g_ptr->feat, CEM_P_CAN_ENTER_PATTERN);
39 p_ptr->riding = old_riding;
40 if (old_pf_ryoute) p_ptr->pet_extra_flags |= (PF_RYOUTE);
41 else p_ptr->pet_extra_flags &= ~(PF_RYOUTE);
42 p_ptr->riding_ryoute = old_riding_ryoute;
43 p_ptr->old_riding_ryoute = old_old_riding_ryoute;
45 p_ptr->update |= PU_BONUS;
48 character_xtra = old_character_xtra;
55 * @brief ペットになっているモンスターをソートするための比較処理
56 * @param u モンスターの構造体配列
58 * @param a 比較対象のモンスターID1
59 * @param b 比較対象のモンスターID2
60 * @return 2番目が大ならばTRUEを返す
62 static bool ang_sort_comp_pet_dismiss(vptr u, vptr v, int a, int b)
64 u16b *who = (u16b*)(u);
69 monster_type *m_ptr1 = ¤t_floor_ptr->m_list[w1];
70 monster_type *m_ptr2 = ¤t_floor_ptr->m_list[w2];
71 monster_race *r_ptr1 = &r_info[m_ptr1->r_idx];
72 monster_race *r_ptr2 = &r_info[m_ptr2->r_idx];
77 if (w1 == p_ptr->riding) return TRUE;
78 if (w2 == p_ptr->riding) return FALSE;
80 if (m_ptr1->nickname && !m_ptr2->nickname) return TRUE;
81 if (m_ptr2->nickname && !m_ptr1->nickname) return FALSE;
83 if (!m_ptr1->parent_m_idx && m_ptr2->parent_m_idx) return TRUE;
84 if (!m_ptr2->parent_m_idx && m_ptr1->parent_m_idx) return FALSE;
86 if ((r_ptr1->flags1 & RF1_UNIQUE) && !(r_ptr2->flags1 & RF1_UNIQUE)) return TRUE;
87 if ((r_ptr2->flags1 & RF1_UNIQUE) && !(r_ptr1->flags1 & RF1_UNIQUE)) return FALSE;
89 if (r_ptr1->level > r_ptr2->level) return TRUE;
90 if (r_ptr2->level > r_ptr1->level) return FALSE;
92 if (m_ptr1->hp > m_ptr2->hp) return TRUE;
93 if (m_ptr2->hp > m_ptr1->hp) return FALSE;
103 int calculate_upkeep(void)
105 s32b old_friend_align = friend_align;
107 bool have_a_unique = FALSE;
108 s32b total_friend_levels = 0;
113 for (m_idx = m_max - 1; m_idx >= 1; m_idx--)
118 m_ptr = ¤t_floor_ptr->m_list[m_idx];
119 if (!monster_is_valid(m_ptr)) continue;
120 r_ptr = &r_info[m_ptr->r_idx];
125 if (r_ptr->flags1 & RF1_UNIQUE)
127 if (p_ptr->pclass == CLASS_CAVALRY)
129 if (p_ptr->riding == m_idx)
130 total_friend_levels += (r_ptr->level + 5) * 2;
131 else if (!have_a_unique && (r_info[m_ptr->r_idx].flags7 & RF7_RIDING))
132 total_friend_levels += (r_ptr->level + 5) * 7 / 2;
134 total_friend_levels += (r_ptr->level + 5) * 10;
135 have_a_unique = TRUE;
138 total_friend_levels += (r_ptr->level + 5) * 10;
141 total_friend_levels += r_ptr->level;
143 /* Determine pet alignment */
144 if (r_ptr->flags3 & RF3_GOOD) friend_align += r_ptr->level;
145 if (r_ptr->flags3 & RF3_EVIL) friend_align -= r_ptr->level;
148 if (old_friend_align != friend_align) p_ptr->update |= (PU_BONUS);
152 upkeep_factor = (total_friend_levels - (p_ptr->lev * 80 / (cp_ptr->pet_upkeep_div)));
153 if (upkeep_factor < 0) upkeep_factor = 0;
154 if (upkeep_factor > 1000) upkeep_factor = 1000;
155 return upkeep_factor;
162 * @brief ペットを開放するコマンドのメインルーチン
165 void do_cmd_pet_dismiss(void)
168 bool all_pets = FALSE;
183 /* Allocate the "who" array */
184 C_MAKE(who, current_floor_ptr->max_m_idx, MONSTER_IDX);
186 /* Process the monsters (backwards) */
187 for (pet_ctr = m_max - 1; pet_ctr >= 1; pet_ctr--)
189 if (is_pet(¤t_floor_ptr->m_list[pet_ctr]))
190 who[max_pet++] = pet_ctr;
193 /* Select the sort method */
194 ang_sort_comp = ang_sort_comp_pet_dismiss;
195 ang_sort_swap = ang_sort_swap_hook;
197 ang_sort(who, &dummy_why, max_pet);
199 /* Process the monsters (backwards) */
200 for (i = 0; i < max_pet; i++)
203 GAME_TEXT friend_name[MAX_NLEN];
206 /* Access the monster */
208 m_ptr = ¤t_floor_ptr->m_list[pet_ctr];
211 kakunin = ((pet_ctr == p_ptr->riding) || (m_ptr->nickname));
212 monster_desc(friend_name, m_ptr, MD_ASSUME_VISIBLE);
216 /* Hack -- health bar for this monster */
217 health_track(pet_ctr);
220 msg_format(_("%sを放しますか? [Yes/No/Unnamed (%d体)]", "Dismiss %s? [Yes/No/Unnamed (%d remain)]"), friend_name, max_pet - i);
223 move_cursor_relative(m_ptr->fy, m_ptr->fx);
229 if (ch == 'Y' || ch == 'y')
235 msg_format(_("本当によろしいですか? (%s) ", "Are you sure? (%s) "), friend_name);
237 if (ch != 'Y' && ch != 'y')
243 if (ch == 'U' || ch == 'u')
249 if (ch == ESCAPE || ch == 'N' || ch == 'n')
256 if ((all_pets && !kakunin) || (!all_pets && delete_this))
258 if (record_named_pet && m_ptr->nickname)
260 GAME_TEXT m_name[MAX_NLEN];
262 monster_desc(m_name, m_ptr, MD_INDEF_VISIBLE);
263 do_cmd_write_nikki(NIKKI_NAMED_PET, RECORD_NAMED_PET_DISMISS, m_name);
266 if (pet_ctr == p_ptr->riding)
268 msg_format(_("%sから降りた。", "You have got off %s. "), friend_name);
272 p_ptr->update |= (PU_BONUS | PU_MONSTERS);
273 p_ptr->redraw |= (PR_EXTRA | PR_UHEALTH);
276 /* HACK : Add the line to message buffer */
277 msg_format(_("%s を放した。", "Dismissed %s."), friend_name);
278 p_ptr->window |= (PW_MESSAGE);
281 delete_monster_idx(pet_ctr);
290 C_KILL(who, current_floor_ptr->max_m_idx, MONSTER_IDX);
293 msg_format("%d 体のペットを放しました。", Dismissed);
295 msg_format("You have dismissed %d pet%s.", Dismissed,
296 (Dismissed == 1 ? "" : "s"));
298 if (Dismissed == 0 && all_pets)
299 msg_print(_("'U'nnamed は、乗馬以外の名前のないペットだけを全て解放します。", "'U'nnamed means all your pets except named pets and your mount."));
305 * @brief ペットから騎乗/下馬するコマンドのメインルーチン /
306 * @param force 強制的に騎乗/下馬するならばTRUE
307 * @return 騎乗/下馬できたらTRUE
309 bool do_riding(bool force)
316 if (!get_direction(&dir, FALSE, FALSE)) return FALSE;
317 y = p_ptr->y + ddy[dir];
318 x = p_ptr->x + ddx[dir];
319 g_ptr = ¤t_floor_ptr->grid_array[y][x];
321 if (p_ptr->special_defense & KATA_MUSOU) set_action(ACTION_NONE);
325 /* Skip non-empty grids */
326 if (!player_can_ride_aux(g_ptr, FALSE))
328 msg_print(_("そちらには降りられません。", "You cannot go to that direction."));
332 if (!pattern_seq(p_ptr->y, p_ptr->x, y, x)) return FALSE;
336 take_turn(p_ptr, 100);
338 msg_print(_("モンスターが立ちふさがっている!", "There is a monster in the way!"));
345 p_ptr->pet_extra_flags &= ~(PF_RYOUTE);
346 p_ptr->riding_ryoute = p_ptr->old_riding_ryoute = FALSE;
350 if (cmd_limit_confused(p_ptr)) return FALSE;
352 m_ptr = ¤t_floor_ptr->m_list[g_ptr->m_idx];
354 if (!g_ptr->m_idx || !m_ptr->ml)
356 msg_print(_("その場所にはモンスターはいません。", "Here is no monster."));
359 if (!is_pet(m_ptr) && !force)
361 msg_print(_("そのモンスターはペットではありません。", "That monster is not a pet."));
364 if (!(r_info[m_ptr->r_idx].flags7 & RF7_RIDING))
366 msg_print(_("そのモンスターには乗れなさそうだ。", "This monster doesn't seem suitable for riding."));
370 if (!pattern_seq(p_ptr->y, p_ptr->x, y, x)) return FALSE;
372 if (!player_can_ride_aux(g_ptr, TRUE))
374 /* Feature code (applying "mimic" field) */
375 feature_type *f_ptr = &f_info[get_feat_mimic(g_ptr)];
377 msg_format("そのモンスターは%sの%sにいる。", f_name + f_ptr->name,
378 ((!have_flag(f_ptr->flags, FF_MOVE) && !have_flag(f_ptr->flags, FF_CAN_FLY)) ||
379 (!have_flag(f_ptr->flags, FF_LOS) && !have_flag(f_ptr->flags, FF_TREE))) ?
382 msg_format("This monster is %s the %s.",
383 ((!have_flag(f_ptr->flags, FF_MOVE) && !have_flag(f_ptr->flags, FF_CAN_FLY)) ||
384 (!have_flag(f_ptr->flags, FF_LOS) && !have_flag(f_ptr->flags, FF_TREE))) ?
385 "in" : "on", f_name + f_ptr->name);
390 if (r_info[m_ptr->r_idx].level > randint1((p_ptr->skill_exp[GINOU_RIDING] / 50 + p_ptr->lev / 2 + 20)))
392 msg_print(_("うまく乗れなかった。", "You failed to ride."));
393 take_turn(p_ptr, 100);
397 if (MON_CSLEEP(m_ptr))
399 GAME_TEXT m_name[MAX_NLEN];
400 monster_desc(m_name, m_ptr, 0);
401 (void)set_monster_csleep(g_ptr->m_idx, 0);
402 msg_format(_("%sを起こした。", "You have waked %s up."), m_name);
405 if (p_ptr->action == ACTION_KAMAE) set_action(ACTION_NONE);
407 p_ptr->riding = g_ptr->m_idx;
409 /* Hack -- remove tracked monster */
410 if (p_ptr->riding == p_ptr->health_who) health_track(0);
413 take_turn(p_ptr, 100);
415 /* Mega-Hack -- Forget the view and lite */
416 p_ptr->update |= (PU_UN_VIEW | PU_UN_LITE);
417 p_ptr->update |= (PU_BONUS);
418 p_ptr->redraw |= (PR_MAP | PR_EXTRA);
419 p_ptr->redraw |= (PR_UHEALTH);
421 (void)move_player_effect(y, x, MPE_HANDLE_STUFF | MPE_ENERGY_USE | MPE_DONT_PICKUP | MPE_DONT_SWAP_MON);
427 * @brief ペットに名前をつけるコマンドのメインルーチン
430 static void do_name_pet(void)
434 GAME_TEXT m_name[MAX_NLEN];
435 bool old_name = FALSE;
436 bool old_target_pet = target_pet;
439 if (!target_set(TARGET_KILL))
441 target_pet = old_target_pet;
444 target_pet = old_target_pet;
446 if (current_floor_ptr->grid_array[target_row][target_col].m_idx)
448 m_ptr = ¤t_floor_ptr->m_list[current_floor_ptr->grid_array[target_row][target_col].m_idx];
452 msg_print(_("そのモンスターはペットではない。", "This monster is not a pet."));
455 if (r_info[m_ptr->r_idx].flags1 & RF1_UNIQUE)
457 msg_print(_("そのモンスターの名前は変えられない!", "You cannot change name of this monster!"));
460 monster_desc(m_name, m_ptr, 0);
462 msg_format(_("%sに名前をつける。", "Name %s."), m_name);
465 /* Start with nothing */
468 /* Use old inscription */
471 /* Start with the old inscription */
472 strcpy(out_val, quark_str(m_ptr->nickname));
476 /* Get a new inscription (possibly empty) */
477 if (get_string(_("名前: ", "Name: "), out_val, 15))
481 /* Save the inscription */
482 m_ptr->nickname = quark_add(out_val);
483 if (record_named_pet)
485 monster_desc(m_name, m_ptr, MD_INDEF_VISIBLE);
486 do_cmd_write_nikki(NIKKI_NAMED_PET, RECORD_NAMED_PET_NAME, m_name);
491 if (record_named_pet && old_name)
493 monster_desc(m_name, m_ptr, MD_INDEF_VISIBLE);
494 do_cmd_write_nikki(NIKKI_NAMED_PET, RECORD_NAMED_PET_UNNAME, m_name);
504 * @brief ペットに関するコマンドリストのメインルーチン /
505 * Issue a pet command
508 void do_cmd_pet(void)
513 concptr power_desc[36];
520 PET_COMMAND_IDX mode = 0;
523 char target_buf[160];
525 int menu_line = use_menu ? 1 : 0;
529 if(p_ptr->wild_mode) return;
531 power_desc[num] = _("ペットを放す", "dismiss pets");
532 powers[num++] = PET_DISMISS;
535 sprintf(target_buf, "ペットのターゲットを指定 (現在:%s)",
536 (pet_t_m_idx ? (p_ptr->image ? "何か奇妙な物" : (r_name + r_info[current_floor_ptr->m_list[pet_t_m_idx].ap_r_idx].name)) : "指定なし"));
538 sprintf(target_buf, "specify a target of pet (now:%s)",
539 (pet_t_m_idx ? (p_ptr->image ? "something strange" : (r_name + r_info[current_floor_ptr->m_list[pet_t_m_idx].ap_r_idx].name)) : "nothing"));
541 power_desc[num] = target_buf;
542 powers[num++] = PET_TARGET;
543 power_desc[num] = _("近くにいろ", "stay close");
545 if (p_ptr->pet_follow_distance == PET_CLOSE_DIST) mode = num;
546 powers[num++] = PET_STAY_CLOSE;
547 power_desc[num] = _("ついて来い", "follow me");
549 if (p_ptr->pet_follow_distance == PET_FOLLOW_DIST) mode = num;
550 powers[num++] = PET_FOLLOW_ME;
551 power_desc[num] = _("敵を見つけて倒せ", "seek and destroy");
553 if (p_ptr->pet_follow_distance == PET_DESTROY_DIST) mode = num;
554 powers[num++] = PET_SEEK_AND_DESTROY;
555 power_desc[num] = _("少し離れていろ", "give me space");
557 if (p_ptr->pet_follow_distance == PET_SPACE_DIST) mode = num;
558 powers[num++] = PET_ALLOW_SPACE;
559 power_desc[num] = _("離れていろ", "stay away");
561 if (p_ptr->pet_follow_distance == PET_AWAY_DIST) mode = num;
562 powers[num++] = PET_STAY_AWAY;
564 if (p_ptr->pet_extra_flags & PF_OPEN_DOORS)
566 power_desc[num] = _("ドアを開ける (現在:ON)", "pets open doors (now On)");
570 power_desc[num] = _("ドアを開ける (現在:OFF)", "pets open doors (now Off)");
572 powers[num++] = PET_OPEN_DOORS;
574 if (p_ptr->pet_extra_flags & PF_PICKUP_ITEMS)
576 power_desc[num] = _("アイテムを拾う (現在:ON)", "pets pick up items (now On)");
580 power_desc[num] = _("アイテムを拾う (現在:OFF)", "pets pick up items (now Off)");
582 powers[num++] = PET_TAKE_ITEMS;
584 if (p_ptr->pet_extra_flags & PF_TELEPORT)
586 power_desc[num] = _("テレポート系魔法を使う (現在:ON)", "allow teleport (now On)");
590 power_desc[num] = _("テレポート系魔法を使う (現在:OFF)", "allow teleport (now Off)");
592 powers[num++] = PET_TELEPORT;
594 if (p_ptr->pet_extra_flags & PF_ATTACK_SPELL)
596 power_desc[num] = _("攻撃魔法を使う (現在:ON)", "allow cast attack spell (now On)");
600 power_desc[num] = _("攻撃魔法を使う (現在:OFF)", "allow cast attack spell (now Off)");
602 powers[num++] = PET_ATTACK_SPELL;
604 if (p_ptr->pet_extra_flags & PF_SUMMON_SPELL)
606 power_desc[num] = _("召喚魔法を使う (現在:ON)", "allow cast summon spell (now On)");
610 power_desc[num] = _("召喚魔法を使う (現在:OFF)", "allow cast summon spell (now Off)");
612 powers[num++] = PET_SUMMON_SPELL;
614 if (p_ptr->pet_extra_flags & PF_BALL_SPELL)
616 power_desc[num] = _("プレイヤーを巻き込む範囲魔法を使う (現在:ON)", "allow involve player in area spell (now On)");
620 power_desc[num] = _("プレイヤーを巻き込む範囲魔法を使う (現在:OFF)", "allow involve player in area spell (now Off)");
622 powers[num++] = PET_BALL_SPELL;
626 power_desc[num] = _("ペットから降りる", "get off a pet");
630 power_desc[num] = _("ペットに乗る", "ride a pet");
632 powers[num++] = PET_RIDING;
633 power_desc[num] = _("ペットに名前をつける", "name pets");
634 powers[num++] = PET_NAME;
638 if ((p_ptr->migite && (empty_hands(FALSE) == EMPTY_HAND_LARM) &&
639 object_allow_two_hands_wielding(&inventory[INVEN_RARM])) ||
640 (p_ptr->hidarite && (empty_hands(FALSE) == EMPTY_HAND_RARM) &&
641 object_allow_two_hands_wielding(&inventory[INVEN_LARM])))
643 if (p_ptr->pet_extra_flags & PF_RYOUTE)
645 power_desc[num] = _("武器を片手で持つ", "use one hand to control a riding pet");
649 power_desc[num] = _("武器を両手で持つ", "use both hands for a weapon");
652 powers[num++] = PET_RYOUTE;
656 switch (p_ptr->pclass)
659 case CLASS_FORCETRAINER:
660 case CLASS_BERSERKER:
661 if (empty_hands(FALSE) == (EMPTY_HAND_RARM | EMPTY_HAND_LARM))
663 if (p_ptr->pet_extra_flags & PF_RYOUTE)
665 power_desc[num] = _("片手で格闘する", "use one hand to control a riding pet");
669 power_desc[num] = _("両手で格闘する", "use both hands for melee");
672 powers[num++] = PET_RYOUTE;
674 else if ((empty_hands(FALSE) != EMPTY_HAND_NONE) && !has_melee_weapon(INVEN_RARM) && !has_melee_weapon(INVEN_LARM))
676 if (p_ptr->pet_extra_flags & PF_RYOUTE)
678 power_desc[num] = _("格闘を行わない", "use one hand to control a riding pet");
682 power_desc[num] = _("格闘を行う", "use one hand for melee");
685 powers[num++] = PET_RYOUTE;
692 if (!(repeat_pull(&i) && (i >= 0) && (i < num)))
694 /* Nothing chosen yet */
705 strnfmt(out_val, 78, _("(コマンド、ESC=終了) コマンドを選んでください:", "(Command, ESC=exit) Choose command from menu."));
711 _("(コマンド %c-%c、'*'=一覧、ESC=終了) コマンドを選んでください:", "(Command %c-%c, *=List, ESC=exit) Select a command: "),
712 I2A(0), I2A(num - 1));
715 choice = (always_show_list || use_menu) ? ESCAPE : 1;
717 /* Get a command from the user */
722 if (choice == ESCAPE) choice = ' ';
723 else if (!get_com(out_val, &choice, TRUE)) break;
725 if (use_menu && (choice != ' '))
736 menu_line += (num - 1);
765 if (menu_line > num) menu_line -= num;
769 if ((choice == ' ') || (choice == '*') || (choice == '?') || (use_menu && ask))
772 if (!redraw || use_menu)
775 PET_COMMAND_IDX ctr = 0;
777 if (!use_menu) screen_save();
782 for (ctr = 0; ctr < num; ctr++)
784 /* Letter/number for power selection */
786 sprintf(buf, "%c%s ", (ctr == mode) ? '*' : ' ', (ctr == (menu_line - 1)) ? _("》", "> ") : " ");
788 sprintf(buf, "%c%c) ", (ctr == mode) ? '*' : ' ', I2A(ctr));
790 strcat(buf, power_desc[ctr]);
792 prt(buf, y + ctr, x);
795 prt("", y + MIN(ctr, 17), x);
813 ask = (isupper(choice));
816 if (ask) choice = (char)tolower(choice);
818 /* Extract request */
819 i = (islower(choice) ? A2I(choice) : -1);
822 /* Totally Illegal */
823 if ((i < 0) || (i >= num))
833 strnfmt(buf, 78, _("%sを使いますか? ", "Use %s? "), power_desc[i]);
835 /* Belay that order */
836 if (!get_check(buf)) continue;
842 if (redraw) screen_load();
844 /* Abort if needed */
855 case PET_DISMISS: /* Dismiss pets */
857 /* Check pets (backwards) */
858 for (pet_ctr = m_max - 1; pet_ctr >= 1; pet_ctr--)
861 if (is_pet(¤t_floor_ptr->m_list[pet_ctr])) break;
866 msg_print(_("ペットがいない!", "You have no pets!"));
869 do_cmd_pet_dismiss();
870 (void)calculate_upkeep();
876 if (!target_set(TARGET_KILL)) pet_t_m_idx = 0;
879 grid_type *g_ptr = ¤t_floor_ptr->grid_array[target_row][target_col];
880 if (g_ptr->m_idx && (current_floor_ptr->m_list[g_ptr->m_idx].ml))
882 pet_t_m_idx = current_floor_ptr->grid_array[target_row][target_col].m_idx;
883 p_ptr->pet_follow_distance = PET_DESTROY_DIST;
885 else pet_t_m_idx = 0;
894 p_ptr->pet_follow_distance = PET_CLOSE_DIST;
901 p_ptr->pet_follow_distance = PET_FOLLOW_DIST;
905 /* "Seek and destoy" */
906 case PET_SEEK_AND_DESTROY:
908 p_ptr->pet_follow_distance = PET_DESTROY_DIST;
911 /* "Give me space" */
912 case PET_ALLOW_SPACE:
914 p_ptr->pet_follow_distance = PET_SPACE_DIST;
920 p_ptr->pet_follow_distance = PET_AWAY_DIST;
923 /* flag - allow pets to open doors */
926 if (p_ptr->pet_extra_flags & PF_OPEN_DOORS) p_ptr->pet_extra_flags &= ~(PF_OPEN_DOORS);
927 else p_ptr->pet_extra_flags |= (PF_OPEN_DOORS);
930 /* flag - allow pets to pickup items */
933 if (p_ptr->pet_extra_flags & PF_PICKUP_ITEMS)
935 p_ptr->pet_extra_flags &= ~(PF_PICKUP_ITEMS);
936 for (pet_ctr = m_max - 1; pet_ctr >= 1; pet_ctr--)
938 /* Access the monster */
939 m_ptr = ¤t_floor_ptr->m_list[pet_ctr];
943 monster_drop_carried_objects(m_ptr);
947 else p_ptr->pet_extra_flags |= (PF_PICKUP_ITEMS);
951 /* flag - allow pets to teleport */
954 if (p_ptr->pet_extra_flags & PF_TELEPORT) p_ptr->pet_extra_flags &= ~(PF_TELEPORT);
955 else p_ptr->pet_extra_flags |= (PF_TELEPORT);
958 /* flag - allow pets to cast attack spell */
959 case PET_ATTACK_SPELL:
961 if (p_ptr->pet_extra_flags & PF_ATTACK_SPELL) p_ptr->pet_extra_flags &= ~(PF_ATTACK_SPELL);
962 else p_ptr->pet_extra_flags |= (PF_ATTACK_SPELL);
965 /* flag - allow pets to cast attack spell */
966 case PET_SUMMON_SPELL:
968 if (p_ptr->pet_extra_flags & PF_SUMMON_SPELL) p_ptr->pet_extra_flags &= ~(PF_SUMMON_SPELL);
969 else p_ptr->pet_extra_flags |= (PF_SUMMON_SPELL);
972 /* flag - allow pets to cast attack spell */
975 if (p_ptr->pet_extra_flags & PF_BALL_SPELL) p_ptr->pet_extra_flags &= ~(PF_BALL_SPELL);
976 else p_ptr->pet_extra_flags |= (PF_BALL_SPELL);
982 (void)do_riding(FALSE);
994 if (p_ptr->pet_extra_flags & PF_RYOUTE) p_ptr->pet_extra_flags &= ~(PF_RYOUTE);
995 else p_ptr->pet_extra_flags |= (PF_RYOUTE);
996 p_ptr->update |= (PU_BONUS);
1005 * @brief ペットの善悪属性に応じた維持コストの途中計算処理
1006 * @param m_ptr 計算基準となるモンスターの構造体参照ポインタ
1007 * @param inc m_ptrで指定したモンスターを維持コスト計算に加えるならTRUE、外すならFALSEを指定
1010 void check_pets_num_and_align(monster_type *m_ptr, bool inc)
1012 s32b old_friend_align = friend_align;
1013 monster_race *r_ptr = &r_info[m_ptr->r_idx];
1018 if (r_ptr->flags3 & RF3_GOOD) friend_align += r_ptr->level;
1019 if (r_ptr->flags3 & RF3_EVIL) friend_align -= r_ptr->level;
1024 if (r_ptr->flags3 & RF3_GOOD) friend_align -= r_ptr->level;
1025 if (r_ptr->flags3 & RF3_EVIL) friend_align += r_ptr->level;
1028 if (old_friend_align != friend_align) p_ptr->update |= (PU_BONUS);
1034 * @brief プレイヤーの落馬判定処理
1035 * @param dam 落馬判定を発した際に受けたダメージ量
1036 * @param force TRUEならば強制的に落馬する
1037 * @return 実際に落馬したらTRUEを返す
1039 bool rakuba(HIT_POINT dam, bool force)
1041 int i, y, x, oy, ox;
1042 int sn = 0, sy = 0, sx = 0;
1043 GAME_TEXT m_name[MAX_NLEN];
1044 monster_type *m_ptr = ¤t_floor_ptr->m_list[p_ptr->riding];
1045 monster_race *r_ptr = &r_info[m_ptr->r_idx];
1046 bool fall_dam = FALSE;
1048 if (!p_ptr->riding) return FALSE;
1049 if (p_ptr->wild_mode) return FALSE;
1051 if (dam >= 0 || force)
1055 int cur = p_ptr->skill_exp[GINOU_RIDING];
1056 int max = s_info[p_ptr->pclass].s_max[GINOU_RIDING];
1057 int ridinglevel = r_ptr->level;
1060 int rakubalevel = r_ptr->level;
1061 if (p_ptr->riding_ryoute) rakubalevel += 20;
1063 if ((cur < max) && (max > 1000) &&
1064 (dam / 2 + ridinglevel) > (cur / 30 + 10))
1068 if (ridinglevel > (cur / 100 + 15))
1069 inc += 1 + (ridinglevel - cur / 100 - 15);
1073 p_ptr->skill_exp[GINOU_RIDING] = MIN(max, cur + inc);
1076 /* レベルの低い乗馬からは落馬しにくい */
1077 if (randint0(dam / 2 + rakubalevel * 2) < cur / 30 + 10)
1079 if ((((p_ptr->pclass == CLASS_BEASTMASTER) || (p_ptr->pclass == CLASS_CAVALRY)) && !p_ptr->riding_ryoute) || !one_in_(p_ptr->lev*(p_ptr->riding_ryoute ? 2 : 3) + 30))
1086 /* Check around the player */
1087 for (i = 0; i < 8; i++)
1091 /* Access the location */
1092 y = p_ptr->y + ddy_ddd[i];
1093 x = p_ptr->x + ddx_ddd[i];
1095 g_ptr = ¤t_floor_ptr->grid_array[y][x];
1097 if (g_ptr->m_idx) continue;
1099 /* Skip non-empty grids */
1100 if (!cave_have_flag_grid(g_ptr, FF_MOVE) && !cave_have_flag_grid(g_ptr, FF_CAN_FLY))
1102 if (!player_can_ride_aux(g_ptr, FALSE)) continue;
1105 if (cave_have_flag_grid(g_ptr, FF_PATTERN)) continue;
1107 /* Count "safe" grids */
1110 /* Randomize choice */
1111 if (randint0(sn) > 0) continue;
1113 /* Save the safe location */
1118 monster_desc(m_name, m_ptr, 0);
1119 msg_format(_("%sから振り落とされそうになって、壁にぶつかった。", "You have nearly fallen from %s, but bumped into wall."), m_name);
1120 take_hit(DAMAGE_NOESCAPE, r_ptr->level + 3, _("壁への衝突", "bumping into wall"), -1);
1130 /* Redraw the old spot */
1133 /* Redraw the new spot */
1134 lite_spot(p_ptr->y, p_ptr->x);
1136 /* Check for new panel */
1141 p_ptr->pet_extra_flags &= ~(PF_RYOUTE);
1142 p_ptr->riding_ryoute = p_ptr->old_riding_ryoute = FALSE;
1144 p_ptr->update |= (PU_BONUS | PU_VIEW | PU_LITE | PU_FLOW | PU_MON_LITE | PU_MONSTERS);
1148 p_ptr->window |= (PW_OVERHEAD | PW_DUNGEON);
1149 p_ptr->redraw |= (PR_EXTRA);
1151 /* Update health track of mount */
1152 p_ptr->redraw |= (PR_UHEALTH);
1154 if (p_ptr->levitation && !force)
1156 monster_desc(m_name, m_ptr, 0);
1157 msg_format(_("%sから落ちたが、空中でうまく体勢を立て直して着地した。", "You are thrown from %s, but make a good landing."), m_name);
1161 take_hit(DAMAGE_NOESCAPE, r_ptr->level + 3, _("落馬", "Falling from riding"), -1);
1165 if (sy && !p_ptr->is_dead)
1166 (void)move_player_effect(p_ptr->y, p_ptr->x, MPE_DONT_PICKUP | MPE_DONT_SWAP_MON);