OSDN Git Service

Merge pull request #796 from shimitei/feature/#791_fix_restore_window_pos
[hengbandforosx/hengbandosx.git] / src / mind / mind-mindcrafter.cpp
1 #include "mind/mind-mindcrafter.h"
2 #include "autopick/autopick.h"
3 #include "core/hp-mp-processor.h"
4 #include "core/player-update-types.h"
5 #include "core/window-redrawer.h"
6 #include "effect/effect-characteristics.h"
7 #include "effect/effect-processor.h"
8 #include "flavor/flag-inscriptions-table.h"
9 #include "flavor/flavor-describer.h"
10 #include "flavor/object-flavor-types.h"
11 #include "floor/floor-object.h"
12 #include "game-option/auto-destruction-options.h"
13 #include "mind/mind-mindcrafter.h"
14 #include "mind/mind-numbers.h"
15 #include "object-enchant/item-feeling.h"
16 #include "object-enchant/special-object-flags.h"
17 #include "object/item-use-flags.h"
18 #include "object/object-mark-types.h"
19 #include "perception/object-perception.h"
20 #include "perception/simple-perception.h"
21 #include "player-info/avatar.h"
22 #include "spell-kind/spells-detection.h"
23 #include "spell-kind/spells-fetcher.h"
24 #include "spell-kind/spells-floor.h"
25 #include "spell-kind/spells-launcher.h"
26 #include "spell-kind/spells-perception.h"
27 #include "spell-kind/spells-sight.h"
28 #include "spell-kind/spells-teleport.h"
29 #include "spell/spell-types.h"
30 #include "spell/spells-status.h"
31 #include "status/bad-status-setter.h"
32 #include "status/buff-setter.h"
33 #include "status/element-resistance.h"
34 #include "status/sight-setter.h"
35 #include "system/object-type-definition.h"
36 #include "target/target-getter.h"
37 #include "util/bit-flags-calculator.h"
38 #include "view/display-messages.h"
39
40 /*!
41  * @brief 超能力者のサイコメトリー処理/ Forcibly pseudo-identify an object in the inventory (or on the floor)
42  * @param caster_ptr プレーヤーへの参照ポインタ
43  * @return なし
44  * @note
45  * currently this function allows pseudo-id of any object,
46  * including silly ones like potions & scrolls, which always
47  * get '{average}'. This should be changed, either to stop such
48  * items from being pseudo-id'd, or to allow psychometry to
49  * detect whether the unidentified potion/scroll/etc is
50  * good (Cure Light Wounds, Restore Strength, etc) or
51  * bad (Poison, Weakness etc) or 'useless' (Slime Mold Juice, etc).
52  */
53 bool psychometry(player_type *caster_ptr)
54 {
55     concptr q = _("どのアイテムを調べますか?", "Meditate on which item? ");
56     concptr s = _("調べるアイテムがありません。", "You have nothing appropriate.");
57     object_type *o_ptr;
58     OBJECT_IDX item;
59     o_ptr = choose_object(caster_ptr, &item, q, s, (USE_EQUIP | USE_INVEN | USE_FLOOR | IGNORE_BOTHHAND_SLOT), TV_NONE);
60     if (!o_ptr)
61         return FALSE;
62
63     if (object_is_known(o_ptr)) {
64         msg_print(_("何も新しいことは判らなかった。", "You cannot find out anything more about that."));
65         return TRUE;
66     }
67
68     item_feel_type feel = pseudo_value_check_heavy(o_ptr);
69     GAME_TEXT o_name[MAX_NLEN];
70     describe_flavor(caster_ptr, o_name, o_ptr, (OD_OMIT_PREFIX | OD_NAME_ONLY));
71
72     if (!feel) {
73         msg_format(_("%sからは特に変わった事は感じとれなかった。", "You do not perceive anything unusual about the %s."), o_name);
74         return TRUE;
75     }
76
77 #ifdef JP
78     msg_format("%sは%sという感じがする...", o_name, game_inscriptions[feel]);
79 #else
80     msg_format("You feel that the %s %s %s...", o_name, ((o_ptr->number == 1) ? "is" : "are"), game_inscriptions[feel]);
81 #endif
82
83     set_bits(o_ptr->ident, IDENT_SENSE);
84     o_ptr->feeling = feel;
85     set_bits(o_ptr->marked, OM_TOUCHED);
86
87     set_bits(caster_ptr->update, PU_COMBINE | PU_REORDER);
88     set_bits(caster_ptr->window_flags, PW_INVEN | PW_EQUIP | PW_PLAYER | PW_FLOOR_ITEM_LIST);
89
90     bool okay = FALSE;
91     switch (o_ptr->tval) {
92     case TV_SHOT:
93     case TV_ARROW:
94     case TV_BOLT:
95     case TV_BOW:
96     case TV_DIGGING:
97     case TV_HAFTED:
98     case TV_POLEARM:
99     case TV_SWORD:
100     case TV_BOOTS:
101     case TV_GLOVES:
102     case TV_HELM:
103     case TV_CROWN:
104     case TV_SHIELD:
105     case TV_CLOAK:
106     case TV_SOFT_ARMOR:
107     case TV_HARD_ARMOR:
108     case TV_DRAG_ARMOR:
109     case TV_CARD:
110     case TV_RING:
111     case TV_AMULET:
112     case TV_LITE:
113     case TV_FIGURINE:
114         okay = TRUE;
115         break;
116
117     default:
118         break;
119     }
120
121     autopick_alter_item(caster_ptr, item, (bool)(okay && destroy_feeling));
122     return TRUE;
123 }
124
125 /*!
126  * @brief 超能力の発動 /
127  * do_cmd_cast calls this function if the player's class is 'mindcrafter'.
128  * @param spell 発動する特殊技能のID
129  * @return 処理を実行したらTRUE、キャンセルした場合FALSEを返す。
130  */
131 bool cast_mindcrafter_spell(player_type *caster_ptr, mind_mindcrafter_type spell)
132 {
133     bool b = FALSE;
134     int dam = 0;
135     DIRECTION dir;
136     TIME_EFFECT t;
137     PLAYER_LEVEL plev = caster_ptr->lev;
138     switch (spell) {
139     case PRECOGNITION:
140         if (plev > 44) {
141             chg_virtue(caster_ptr, V_KNOWLEDGE, 1);
142             chg_virtue(caster_ptr, V_ENLIGHTEN, 1);
143             wiz_lite(caster_ptr, FALSE);
144         } else if (plev > 19)
145             map_area(caster_ptr, DETECT_RAD_MAP);
146
147         if (plev < 30) {
148             b = detect_monsters_normal(caster_ptr, DETECT_RAD_DEFAULT);
149             if (plev > 14)
150                 b |= detect_monsters_invis(caster_ptr, DETECT_RAD_DEFAULT);
151             if (plev > 4) {
152                 b |= detect_traps(caster_ptr, DETECT_RAD_DEFAULT, TRUE);
153                 b |= detect_doors(caster_ptr, DETECT_RAD_DEFAULT);
154             }
155         } else {
156             b = detect_all(caster_ptr, DETECT_RAD_DEFAULT);
157         }
158
159         if ((plev > 24) && (plev < 40))
160             set_tim_esp(caster_ptr, (TIME_EFFECT)plev, FALSE);
161
162         if (!b)
163             msg_print(_("安全な気がする。", "You feel safe."));
164
165         break;
166     case NEURAL_BLAST:
167         if (!get_aim_dir(caster_ptr, &dir))
168             return FALSE;
169
170         if (randint1(100) < plev * 2)
171             fire_beam(caster_ptr, GF_PSI, dir, damroll(3 + ((plev - 1) / 4), (3 + plev / 15)));
172         else
173             fire_ball(caster_ptr, GF_PSI, dir, damroll(3 + ((plev - 1) / 4), (3 + plev / 15)), 0);
174         break;
175     case MINOR_DISPLACEMENT:
176         teleport_player(caster_ptr, 10, TELEPORT_SPONTANEOUS);
177         break;
178     case MAJOR_DISPLACEMENT:
179         teleport_player(caster_ptr, plev * 5, TELEPORT_SPONTANEOUS);
180         break;
181     case DOMINATION:
182         if (plev < 30) {
183             if (!get_aim_dir(caster_ptr, &dir))
184                 return FALSE;
185
186             fire_ball(caster_ptr, GF_DOMINATION, dir, plev, 0);
187         } else {
188             charm_monsters(caster_ptr, plev * 2);
189         }
190
191         break;
192     case PLUVERISE:
193         if (!get_aim_dir(caster_ptr, &dir))
194             return FALSE;
195
196         fire_ball(caster_ptr, GF_TELEKINESIS, dir, damroll(8 + ((plev - 5) / 4), 8), (plev > 20 ? (plev - 20) / 8 + 1 : 0));
197         break;
198     case CHARACTER_ARMOR:
199         set_shield(caster_ptr, (TIME_EFFECT)plev, FALSE);
200         if (plev > 14)
201             set_oppose_acid(caster_ptr, (TIME_EFFECT)plev, FALSE);
202         if (plev > 19)
203             set_oppose_fire(caster_ptr, (TIME_EFFECT)plev, FALSE);
204         if (plev > 24)
205             set_oppose_cold(caster_ptr, (TIME_EFFECT)plev, FALSE);
206         if (plev > 29)
207             set_oppose_elec(caster_ptr, (TIME_EFFECT)plev, FALSE);
208         if (plev > 34)
209             set_oppose_pois(caster_ptr, (TIME_EFFECT)plev, FALSE);
210
211         break;
212     case PSYCHOMETRY:
213         if (plev < 25)
214             return psychometry(caster_ptr);
215         else
216             return ident_spell(caster_ptr, FALSE, TV_NONE);
217     case MIND_WAVE:
218         msg_print(_("精神を捻じ曲げる波動を発生させた!", "Mind-warping forces emanate from your brain!"));
219         if (plev < 25)
220             project(caster_ptr, 0, 2 + plev / 10, caster_ptr->y, caster_ptr->x, (plev * 3), GF_PSI, PROJECT_KILL);
221         else
222             (void)mindblast_monsters(caster_ptr, randint1(plev * ((plev - 5) / 10 + 1)));
223
224         break;
225     case ADRENALINE_CHANNELING:
226         set_afraid(caster_ptr, 0);
227         set_stun(caster_ptr, 0);
228         if (!is_fast(caster_ptr) || !is_hero(caster_ptr))
229             hp_player(caster_ptr, plev);
230
231         t = 10 + randint1((plev * 3) / 2);
232         set_hero(caster_ptr, t, FALSE);
233         (void)set_fast(caster_ptr, t, FALSE);
234         break;
235     case TELEKINESIS:
236         if (!get_aim_dir(caster_ptr, &dir))
237             return FALSE;
238
239         fetch_item(caster_ptr, dir, plev * 15, FALSE);
240         break;
241     case PSYCHIC_DRAIN:
242         if (!get_aim_dir(caster_ptr, &dir))
243             return FALSE;
244
245         dam = damroll(plev / 2, 6);
246         if (fire_ball(caster_ptr, GF_PSI_DRAIN, dir, dam, 0))
247             caster_ptr->energy_need += randint1(150);
248
249         break;
250     case PSYCHO_SPEAR:
251         if (!get_aim_dir(caster_ptr, &dir))
252             return FALSE;
253
254         fire_beam(caster_ptr, GF_PSY_SPEAR, dir, randint1(plev * 3) + plev * 3);
255         break;
256     case THE_WORLD:
257         time_walk(caster_ptr);
258         break;
259     default:
260         msg_print(_("なに?", "Zap?"));
261     }
262
263     return TRUE;
264 }