OSDN Git Service

Merge branch 'master' of git.sourceforge.jp:/gitroot/hengband/hengband
[hengband/hengband.git] / src / hissatsu.c
1 /*!
2  * @file hissatsu.c
3  * @brief 剣術の実装 / Blade arts
4  * @date 2014/01/17
5  * @author
6  * Copyright (c) 1997 Ben Harrison, James E. Wilson, Robert A. Koeneke\n
7  * This software may be copied and distributed for educational, research,\n
8  * and not for profit purposes provided that this copyright and statement\n
9  * are included in all such copies.  Other copyrights may also apply.\n
10  * 2014 Deskull rearranged comment for Doxygen.\n
11  */
12
13 #include "angband.h"
14
15 #define TECHNIC_HISSATSU (REALM_HISSATSU - MIN_TECHNIC)
16
17
18 /*!
19  * @brief 使用可能な剣術を選択する /
20  * Allow user to choose a blade arts.
21  * @param sn 選択した特殊技能ID、キャンセルの場合-1、不正な選択の場合-2を返す
22  * @return 発動可能な魔法を選択した場合TRUE、キャンセル処理か不正な選択が行われた場合FALSEを返す。
23  * @details
24  * If a valid spell is chosen, saves it in '*sn' and returns TRUE\n
25  * If the user hits escape, returns FALSE, and set '*sn' to -1\n
26  * If there are no legal choices, returns FALSE, and sets '*sn' to -2\n
27  *\n
28  * The "prompt" should be "cast", "recite", or "study"\n
29  * The "known" should be TRUE for cast/pray, FALSE for study\n
30  *\n
31  * nb: This function has a (trivial) display bug which will be obvious\n
32  * when you run it. It's probably easy to fix but I haven't tried,\n
33  * sorry.\n
34  */
35 static int get_hissatsu_power(SPELL_IDX *sn)
36 {
37         SPELL_IDX i;
38         int j = 0;
39         int             num = 0;
40         int             y = 1;
41         int             x = 15;
42         int             plev = p_ptr->lev;
43         int             ask = TRUE;
44         char            choice;
45         char            out_val[160];
46         SPELL_IDX sentaku[32];
47         cptr            p = _("必殺剣", "special attack");
48         COMMAND_CODE code;
49         magic_type spell;
50         bool            flag, redraw;
51         int menu_line = (use_menu ? 1 : 0);
52
53         /* Assume cancelled */
54         *sn = (-1);
55
56 #ifdef ALLOW_REPEAT /* TNB */
57
58         /* Get the spell, if available */
59         if (repeat_pull(&code))
60         {
61                 *sn = (SPELL_IDX)code;
62                 /* Verify the spell */
63                 if (technic_info[TECHNIC_HISSATSU][*sn].slevel <= plev)
64                 {
65                         /* Success */
66                         return (TRUE);
67                 }
68         }
69
70 #endif /* ALLOW_REPEAT -- TNB */
71
72         /* Nothing chosen yet */
73         flag = FALSE;
74
75         /* No redraw yet */
76         redraw = FALSE;
77
78         for (i = 0; i < 32; i++)
79         {
80                 if (technic_info[TECHNIC_HISSATSU][i].slevel <= PY_MAX_LEVEL)
81                 {
82                         sentaku[num] = i;
83                         num++;
84                 }
85         }
86
87         /* Build a prompt (accept all spells) */
88         (void) strnfmt(out_val, 78, 
89                        _("(%^s %c-%c, '*'で一覧, ESC) どの%sを使いますか?", "(%^ss %c-%c, *=List, ESC=exit) Use which %s? "),
90                        p, I2A(0), "abcdefghijklmnopqrstuvwxyz012345"[num-1], p);
91
92         if (use_menu) screen_save();
93
94         /* Get a spell from the user */
95
96         choice= always_show_list ? ESCAPE:1 ;
97         while (!flag)
98         {
99                 if(choice==ESCAPE) choice = ' '; 
100                 else if( !get_com(out_val, &choice, FALSE) )break;
101
102                 if (use_menu && choice != ' ')
103                 {
104                         switch(choice)
105                         {
106                                 case '0':
107                                 {
108                                         screen_load();
109                                         return (FALSE);
110                                 }
111
112                                 case '8':
113                                 case 'k':
114                                 case 'K':
115                                 {
116                                         do
117                                         {
118                                                 menu_line += 31;
119                                                 if (menu_line > 32) menu_line -= 32;
120                                         } while(!(p_ptr->spell_learned1 & (1L << (menu_line-1))));
121                                         break;
122                                 }
123
124                                 case '2':
125                                 case 'j':
126                                 case 'J':
127                                 {
128                                         do
129                                         {
130                                                 menu_line++;
131                                                 if (menu_line > 32) menu_line -= 32;
132                                         } while(!(p_ptr->spell_learned1 & (1L << (menu_line-1))));
133                                         break;
134                                 }
135
136                                 case '4':
137                                 case 'h':
138                                 case 'H':
139                                 case '6':
140                                 case 'l':
141                                 case 'L':
142                                 {
143                                         bool reverse = FALSE;
144                                         if ((choice == '4') || (choice == 'h') || (choice == 'H')) reverse = TRUE;
145                                         if (menu_line > 16)
146                                         {
147                                                 menu_line -= 16;
148                                                 reverse = TRUE;
149                                         }
150                                         else menu_line+=16;
151                                         while(!(p_ptr->spell_learned1 & (1L << (menu_line-1))))
152                                         {
153                                                 if (reverse)
154                                                 {
155                                                         menu_line--;
156                                                         if (menu_line < 2) reverse = FALSE;
157                                                 }
158                                                 else
159                                                 {
160                                                         menu_line++;
161                                                         if (menu_line > 31) reverse = TRUE;
162                                                 }
163                                         }
164                                         break;
165                                 }
166
167                                 case 'x':
168                                 case 'X':
169                                 case '\r':
170                                 case '\n':
171                                 {
172                                         i = menu_line - 1;
173                                         ask = FALSE;
174                                         break;
175                                 }
176                         }
177                 }
178                 /* Request redraw */
179                 if ((choice == ' ') || (choice == '*') || (choice == '?') || (use_menu && ask))
180                 {
181                         /* Show the list */
182                         if (!redraw || use_menu)
183                         {
184                                 char psi_desc[80];
185                                 int line;
186
187                                 /* Show list */
188                                 redraw = TRUE;
189
190                                 /* Save the screen */
191                                 if (!use_menu) screen_save();
192
193                                 /* Display a list of spells */
194                                 prt("", y, x);
195                                 put_str(_("名前              Lv  MP      名前              Lv  MP ", 
196                                                   "name              Lv  SP      name              Lv  SP "), y, x + 5);
197                                 prt("", y+1, x);
198                                 /* Dump the spells */
199                                 for (i = 0, line = 0; i < 32; i++)
200                                 {
201                                         spell = technic_info[TECHNIC_HISSATSU][i];
202
203                                         if (spell.slevel > PY_MAX_LEVEL) continue;
204                                         line++;
205                                         if (!(p_ptr->spell_learned1 >> i)) break;
206
207                                         /* Access the spell */
208                                         if (spell.slevel > plev)   continue;
209                                         if (!(p_ptr->spell_learned1 & (1L << i))) continue;
210                                         if (use_menu)
211                                         {
212                                                 if (i == (menu_line-1))
213                                                         strcpy(psi_desc, _("  》", "  > "));
214                                                 else strcpy(psi_desc, "    ");
215                                                 
216                                         }
217                                         else
218                                         {
219                                                 char letter;
220                                                 if (line <= 26)
221                                                         letter = I2A(line-1);
222                                                 else
223                                                         letter = '0' + line - 27;
224                                                 sprintf(psi_desc, "  %c)",letter);
225                                         }
226
227                                         /* Dump the spell --(-- */
228                                         strcat(psi_desc, format(" %-18s%2d %3d",
229                                                 do_spell(REALM_HISSATSU, i, SPELL_NAME),
230                                                 spell.slevel, spell.smana));
231                                         prt(psi_desc, y + (line%17) + (line >= 17), x+(line/17)*30);
232                                         prt("", y + (line%17) + (line >= 17) + 1, x+(line/17)*30);
233                                 }
234                         }
235
236                         /* Hide the list */
237                         else
238                         {
239                                 /* Hide list */
240                                 redraw = FALSE;
241
242                                 /* Restore the screen */
243                                 screen_load();
244                         }
245
246                         /* Redo asking */
247                         continue;
248                 }
249
250                 if (!use_menu)
251                 {
252                         if (isalpha(choice))
253                         {
254                                 /* Note verify */
255                                 ask = (isupper(choice));
256
257                                 /* Lowercase */
258                                 if (ask) choice = (char)tolower(choice);
259
260                                 /* Extract request */
261                                 i = (islower(choice) ? A2I(choice) : -1);
262                         }
263                         else
264                         {
265                                 ask = FALSE; /* Can't uppercase digits */
266
267                                 i = choice - '0' + 26;
268                         }
269                 }
270
271                 /* Totally Illegal */
272                 if ((i < 0) || (i >= 32) || !(p_ptr->spell_learned1 & (1 << sentaku[i])))
273                 {
274                         bell();
275                         continue;
276                 }
277
278                 j = sentaku[i];
279
280                 /* Verify it */
281                 if (ask)
282                 {
283                         char tmp_val[160];
284
285                         /* Prompt */
286                         (void) strnfmt(tmp_val, 78, _("%sを使いますか?", "Use %s? "), do_spell(REALM_HISSATSU, j, SPELL_NAME));
287
288                         /* Belay that order */
289                         if (!get_check(tmp_val)) continue;
290                 }
291
292                 /* Stop the loop */
293                 flag = TRUE;
294         }
295
296         /* Restore the screen */
297         if (redraw) screen_load();
298
299         /* Show choices */
300         p_ptr->window |= (PW_SPELL);
301
302         /* Window stuff */
303         window_stuff();
304
305
306         /* Abort if needed */
307         if (!flag) return (FALSE);
308
309         /* Save the choice */
310         (*sn) = j;
311
312 #ifdef ALLOW_REPEAT /* TNB */
313
314         repeat_push((COMMAND_CODE)j);
315
316 #endif /* ALLOW_REPEAT -- TNB */
317
318         /* Success */
319         return (TRUE);
320 }
321
322
323 /*!
324  * @brief 剣術コマンドのメインルーチン
325  * @return なし
326  */
327 void do_cmd_hissatsu(void)
328 {
329         SPELL_IDX       n = 0;
330         magic_type      spell;
331
332
333         /* not if confused */
334         if (p_ptr->confused)
335         {
336                 msg_print(_("混乱していて集中できない!", "You are too confused!"));
337                 return;
338         }
339         if (!buki_motteruka(INVEN_RARM) && !buki_motteruka(INVEN_LARM))
340         {
341                 if (flush_failure) flush();
342                 msg_print(_("武器を持たないと必殺技は使えない!", "You need to wield a weapon!"));
343                 return;
344         }
345         if (!p_ptr->spell_learned1)
346         {
347                 msg_print(_("何も技を知らない。", "You don't know any special attacks."));
348                 return;
349         }
350
351         if (p_ptr->special_defense & KATA_MASK)
352         {
353                 set_action(ACTION_NONE);
354         }
355
356         /* get power */
357         if (!get_hissatsu_power(&n)) return;
358
359         spell = technic_info[TECHNIC_HISSATSU][n];
360
361         /* Verify "dangerous" spells */
362         if (spell.smana > p_ptr->csp)
363         {
364                 if (flush_failure) flush();
365                 /* Warning */
366                 msg_print(_("MPが足りません。", "You do not have enough mana to use this power."));
367                 msg_print(NULL);
368                 return;
369         }
370
371         sound(SOUND_ZAP);
372
373         /* Cast the spell */
374         if (!do_spell(REALM_HISSATSU, n, SPELL_CAST)) return;
375
376         /* Take a turn */
377         p_ptr->energy_use = 100;
378
379         /* Use some mana */
380         p_ptr->csp -= spell.smana;
381
382         /* Limit */
383         if (p_ptr->csp < 0) p_ptr->csp = 0;
384
385         /* Redraw mana */
386         p_ptr->redraw |= (PR_MANA);
387
388         /* Window stuff */
389         p_ptr->window |= (PW_PLAYER);
390         p_ptr->window |= (PW_SPELL);
391 }
392
393
394 /*!
395  * @brief 剣術コマンドの学習
396  * @return なし
397  */
398 void do_cmd_gain_hissatsu(void)
399 {
400         OBJECT_IDX item;
401         int i, j;
402
403         object_type *o_ptr;
404         cptr q, s;
405
406         bool gain = FALSE;
407
408         if (p_ptr->special_defense & (KATA_MUSOU | KATA_KOUKIJIN))
409         {
410                 set_action(ACTION_NONE);
411         }
412
413         if (p_ptr->blind || no_lite())
414         {
415                 msg_print(_("目が見えない!", "You cannot see!"));
416                 return;
417         }
418
419         if (p_ptr->confused)
420         {
421                 msg_print(_("混乱していて読めない!", "You are too confused!"));
422                 return;
423         }
424
425         if (!(p_ptr->new_spells))
426         {
427                 msg_print(_("新しい必殺技を覚えることはできない!", "You cannot learn any new special attacks!"));
428                 return;
429         }
430
431 #ifdef JP
432         if( p_ptr->new_spells < 10 ){
433                 msg_format("あと %d つの必殺技を学べる。", p_ptr->new_spells);
434         }else{
435                 msg_format("あと %d 個の必殺技を学べる。", p_ptr->new_spells);
436         }
437 #else
438         msg_format("You can learn %d new special attack%s.", p_ptr->new_spells,
439                 (p_ptr->new_spells == 1?"":"s"));
440 #endif
441
442         item_tester_tval = TV_HISSATSU_BOOK;
443
444         /* Get an item */
445         q = _("どの書から学びますか? ", "Study which book? ");
446         s = _("読める書がない。", "You have no books that you can read.");
447
448         if (!get_item(&item, q, s, (USE_INVEN | USE_FLOOR))) return;
449
450         /* Get the item (in the pack) */
451         if (item >= 0)
452         {
453                 o_ptr = &inventory[item];
454         }
455
456         /* Get the item (on the floor) */
457         else
458         {
459                 o_ptr = &o_list[0 - item];
460         }
461
462         for (i = o_ptr->sval * 8; i < o_ptr->sval * 8 + 8; i++)
463         {
464                 if (p_ptr->spell_learned1 & (1L << i)) continue;
465                 if (technic_info[TECHNIC_HISSATSU][i].slevel > p_ptr->lev) continue;
466
467                 p_ptr->spell_learned1 |= (1L << i);
468                 p_ptr->spell_worked1 |= (1L << i);
469                 msg_format(_("%sの技を覚えた。", "You have learned the special attack of %s."), do_spell(REALM_HISSATSU, i, SPELL_NAME));
470                 for (j = 0; j < 64; j++)
471                 {
472                         /* Stop at the first empty space */
473                         if (p_ptr->spell_order[j] == 99) break;
474                 }
475                 p_ptr->spell_order[j] = i;
476                 gain = TRUE;
477         }
478
479         /* No gain ... */
480         if (!gain)
481                 msg_print(_("何も覚えられなかった。", "You were not able to learn any special attacks."));
482
483         /* Take a turn */
484         else
485                 p_ptr->energy_use = 100;
486
487         p_ptr->update |= (PU_SPELLS);
488 }
489
490
491 /*!
492  * @brief 剣術のスレイ倍率計算を行う /
493  * Calcurate magnification of hissatsu technics
494  * @param mult 剣術のスレイ効果以前に算出している多要素の倍率(/10倍)
495  * @param flgs 剣術に使用する武器のスレイフラグ配列
496  * @param m_ptr 目標となるモンスターの構造体参照ポインタ
497  * @param mode 剣術のスレイ型ID
498  * @return スレイの倍率(/10倍)
499  */
500 MULTIPLY mult_hissatsu(MULTIPLY mult, BIT_FLAGS *flgs, monster_type *m_ptr, BIT_FLAGS mode)
501 {
502         monster_race *r_ptr = &r_info[m_ptr->r_idx];
503
504         /* Burning Strike (Fire) */
505         if (mode == HISSATSU_FIRE)
506         {
507                 /* Notice immunity */
508                 if (r_ptr->flagsr & RFR_EFF_IM_FIRE_MASK)
509                 {
510                         if (is_original_ap_and_seen(m_ptr))
511                         {
512                                 r_ptr->r_flagsr |= (r_ptr->flagsr & RFR_EFF_IM_FIRE_MASK);
513                         }
514                 }
515
516                 /* Otherwise, take the damage */
517                 else if (have_flag(flgs, TR_BRAND_FIRE))
518                 {
519                         if (r_ptr->flags3 & RF3_HURT_FIRE)
520                         {
521                                 if (mult < 70) mult = 70;
522                                 if (is_original_ap_and_seen(m_ptr))
523                                 {
524                                         r_ptr->r_flags3 |= RF3_HURT_FIRE;
525                                 }
526                         }
527                         else if (mult < 35) mult = 35;
528                 }
529                 else
530                 {
531                         if (r_ptr->flags3 & RF3_HURT_FIRE)
532                         {
533                                 if (mult < 50) mult = 50;
534                                 if (is_original_ap_and_seen(m_ptr))
535                                 {
536                                         r_ptr->r_flags3 |= RF3_HURT_FIRE;
537                                 }
538                         }
539                         else if (mult < 25) mult = 25;
540                 }
541         }
542
543         /* Serpent's Tongue (Poison) */
544         if (mode == HISSATSU_POISON)
545         {
546                 /* Notice immunity */
547                 if (r_ptr->flagsr & RFR_EFF_IM_POIS_MASK)
548                 {
549                         if (is_original_ap_and_seen(m_ptr))
550                         {
551                                 r_ptr->r_flagsr |= (r_ptr->flagsr & RFR_EFF_IM_POIS_MASK);
552                         }
553                 }
554
555                 /* Otherwise, take the damage */
556                 else if (have_flag(flgs, TR_BRAND_POIS))
557                 {
558                         if (mult < 35) mult = 35;
559                 }
560                 else
561                 {
562                         if (mult < 25) mult = 25;
563                 }
564         }
565
566         /* Zammaken (Nonliving Evil) */
567         if (mode == HISSATSU_ZANMA)
568         {
569                 if (!monster_living(r_ptr) && (r_ptr->flags3 & RF3_EVIL))
570                 {
571                         if (mult < 15) mult = 25;
572                         else if (mult < 50) mult = MIN(50, mult+20);
573                 }
574         }
575
576         /* Rock Smash (Hurt Rock) */
577         if (mode == HISSATSU_HAGAN)
578         {
579                 if (r_ptr->flags3 & RF3_HURT_ROCK)
580                 {
581                         if (is_original_ap_and_seen(m_ptr))
582                         {
583                                 r_ptr->r_flags3 |= RF3_HURT_ROCK;
584                         }
585                         if (mult == 10) mult = 40;
586                         else if (mult < 60) mult = 60;
587                 }
588         }
589
590         /* Midare-Setsugekka (Cold) */
591         if (mode == HISSATSU_COLD)
592         {
593                 /* Notice immunity */
594                 if (r_ptr->flagsr & RFR_EFF_IM_COLD_MASK)
595                 {
596                         if (is_original_ap_and_seen(m_ptr))
597                         {
598                                 r_ptr->r_flagsr |= (r_ptr->flagsr & RFR_EFF_IM_COLD_MASK);
599                         }
600                 }
601                 /* Otherwise, take the damage */
602                 else if (have_flag(flgs, TR_BRAND_COLD))
603                 {
604                         if (r_ptr->flags3 & RF3_HURT_COLD)
605                         {
606                                 if (mult < 70) mult = 70;
607                                 if (is_original_ap_and_seen(m_ptr))
608                                 {
609                                         r_ptr->r_flags3 |= RF3_HURT_COLD;
610                                 }
611                         }
612                         else if (mult < 35) mult = 35;
613                 }
614                 else
615                 {
616                         if (r_ptr->flags3 & RF3_HURT_COLD)
617                         {
618                                 if (mult < 50) mult = 50;
619                                 if (is_original_ap_and_seen(m_ptr))
620                                 {
621                                         r_ptr->r_flags3 |= RF3_HURT_COLD;
622                                 }
623                         }
624                         else if (mult < 25) mult = 25;
625                 }
626         }
627
628         /* Lightning Eagle (Elec) */
629         if (mode == HISSATSU_ELEC)
630         {
631                 /* Notice immunity */
632                 if (r_ptr->flagsr & RFR_EFF_IM_ELEC_MASK)
633                 {
634                         if (is_original_ap_and_seen(m_ptr))
635                         {
636                                 r_ptr->r_flagsr |= (r_ptr->flagsr & RFR_EFF_IM_ELEC_MASK);
637                         }
638                 }
639
640                 /* Otherwise, take the damage */
641                 else if (have_flag(flgs, TR_BRAND_ELEC))
642                 {
643                         if (mult < 70) mult = 70;
644                 }
645                 else
646                 {
647                         if (mult < 50) mult = 50;
648                 }
649         }
650
651         /* Bloody Maelstrom */
652         if ((mode == HISSATSU_SEKIRYUKA) && p_ptr->cut && monster_living(r_ptr))
653         {
654                 MULTIPLY tmp = MIN(100, MAX(10, p_ptr->cut / 10));
655                 if (mult < tmp) mult = tmp;
656         }
657
658         /* Keiun-Kininken */
659         if (mode == HISSATSU_UNDEAD)
660         {
661                 if (r_ptr->flags3 & RF3_UNDEAD)
662                 {
663                         if (is_original_ap_and_seen(m_ptr))
664                         {
665                                 r_ptr->r_flags3 |= RF3_UNDEAD;
666                         }
667                         if (mult == 10) mult = 70;
668                         else if (mult < 140) mult = MIN(140, mult+60);
669                 }
670                 if (mult == 10) mult = 40;
671                 else if (mult < 60) mult = MIN(60, mult+30);
672         }
673
674         if (mult > 150) mult = 150;
675
676         return mult;
677 }