]>
git.cameronkatri.com Git - bsdgames-darwin.git/blob - hunt/huntd/shots.c
1 /* $NetBSD: shots.c,v 1.16 2021/05/02 12:50:45 rillig Exp $ */
3 * Copyright (c) 1983-2003, Regents of the University of California.
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are
10 * + Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * + Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * + Neither the name of the University of California, San Francisco nor
16 * the names of its contributors may be used to endorse or promote
17 * products derived from this software without specific prior written
20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
21 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
22 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
23 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33 #include <sys/cdefs.h>
35 __RCSID("$NetBSD: shots.c,v 1.16 2021/05/02 12:50:45 rillig Exp $");
43 #define PLUS_DELTA(x, max) if (x < max) x++; else x--
44 #define MINUS_DELTA(x, min) if (x > min) x--; else x++
46 static void chkshot(BULLET
*, BULLET
*);
47 static void chkslime(BULLET
*, BULLET
*);
48 static void explshot(BULLET
*, int, int);
49 static void find_under(BULLET
*, BULLET
*);
50 static bool iswall(int, int);
51 static void mark_boot(BULLET
*);
52 static void mark_player(BULLET
*);
54 static void move_drone(BULLET
*);
56 static void move_flyer(PLAYER
*);
57 static int move_normal_shot(BULLET
*);
58 static void move_slime(BULLET
*, int, BULLET
*);
59 static void save_bullet(BULLET
*);
60 static void zapshot(BULLET
*, BULLET
*);
64 * Move the shots already in the air, taking explosions into account
79 * First we move through the bullet list BULSPD times, looking
80 * for things we may have run into. If we do run into
81 * something, we set up the explosion and disappear, checking
82 * for damage to any player who got in the way.
87 for (bp
= blist
; bp
!= NULL
; bp
= next
) {
91 Maze
[y
][x
] = bp
->b_over
;
92 for (pp
= Player
; pp
< End_player
; pp
++)
95 for (pp
= Monitor
; pp
< End_monitor
; pp
++)
104 if (move_normal_shot(bp
)) {
105 bp
->b_next
= Bullets
;
111 if (bp
->b_expl
|| move_normal_shot(bp
)) {
112 bp
->b_next
= Bullets
;
119 if (move_drone(bp
)) {
120 bp
->b_next
= Bullets
;
126 bp
->b_next
= Bullets
;
134 for (bp
= blist
; bp
!= NULL
; bp
= next
) {
139 for (pp
= Monitor
; pp
< End_monitor
; pp
++)
140 check(pp
, bp
->b_y
, bp
->b_x
);
143 if (bp
->b_type
== DSHOT
)
144 for (pp
= Player
; pp
< End_player
; pp
++)
146 check(pp
, bp
->b_y
, bp
->b_x
);
155 for (pp
= Player
; pp
< End_player
; pp
++)
156 Maze
[pp
->p_y
][pp
->p_x
] = pp
->p_face
;
160 for (pp
= Boot
; pp
< &Boot
[NBOOTS
]; pp
++)
161 if (pp
->p_flying
>= 0)
164 for (pp
= Player
; pp
< End_player
; pp
++) {
166 if (pp
->p_flying
>= 0)
169 sendcom(pp
, REFRESH
); /* Flush out the explosions */
171 sendcom(pp
, REFRESH
);
174 for (pp
= Monitor
; pp
< End_monitor
; pp
++)
175 sendcom(pp
, REFRESH
);
183 * Move a normal shot along its trajectory
186 move_normal_shot(BULLET
*bp
)
191 for (i
= 0; i
< BULSPD
; i
++) {
198 switch (bp
->b_face
) {
213 switch (Maze
[y
][x
]) {
215 if (rand_num(100) < 5) {
216 zapshot(Bullets
, bp
);
217 zapshot(bp
->b_next
, bp
);
221 if (rand_num(100) < 10) {
222 zapshot(Bullets
, bp
);
223 zapshot(bp
->b_next
, bp
);
227 case WALL4
: /* reflecting walls */
228 switch (bp
->b_face
) {
244 for (pp
= Monitor
; pp
< End_monitor
; pp
++)
249 switch (bp
->b_face
) {
265 for (pp
= Monitor
; pp
< End_monitor
; pp
++)
272 switch (rand_num(4)) {
291 message(pp
, "Zing!");
299 * give the person a chance to catch a
300 * grenade if s/he is facing it
303 pp
->p_ident
->i_shot
+= bp
->b_charge
;
304 if (opposite(bp
->b_face
, Maze
[y
][x
])) {
305 if (rand_num(100) < 10) {
306 if (bp
->b_owner
!= NULL
)
308 "Your charge was absorbed!");
309 if (bp
->b_score
!= NULL
)
310 bp
->b_score
->i_robbed
+= bp
->b_charge
;
311 pp
->p_ammo
+= bp
->b_charge
;
312 if (pp
->p_damage
+ bp
->b_size
* MINDAM
314 pp
->p_ident
->i_saved
++;
315 message(pp
, "Absorbed charge (good shield!)");
316 pp
->p_ident
->i_absorbed
+= bp
->b_charge
;
318 (void) snprintf(Buf
, sizeof(Buf
),
320 cgoto(pp
, STAT_AMMO_ROW
, STAT_VALUE_COL
);
324 pp
->p_ident
->i_faced
+= bp
->b_charge
;
327 * Small chance that the bullet just misses the
328 * person. If so, the bullet just goes on its
329 * merry way without exploding.
331 if (rand_num(100) < 5) {
332 pp
->p_ident
->i_ducked
+= bp
->b_charge
;
333 if (pp
->p_damage
+ bp
->b_size
* MINDAM
335 pp
->p_ident
->i_saved
++;
336 if (bp
->b_score
!= NULL
)
337 bp
->b_score
->i_missed
+= bp
->b_charge
;
338 message(pp
, "Zing!");
339 if (bp
->b_owner
== NULL
)
341 message(bp
->b_owner
, bp
->b_score
&&
342 ((bp
->b_score
->i_missed
& 0x7) == 0x7) ?
343 "My! What a bad shot you are!" :
348 * The shot hit that sucker! Blow it up.
371 * Move the drone to the next square
374 move_drone(BULLET
*bp
)
381 * See if we can give someone a blast
383 if (isplayer(Maze
[bp
->b_y
][bp
->b_x
- 1])) {
387 if (isplayer(Maze
[bp
->b_y
- 1][bp
->b_x
])) {
391 if (isplayer(Maze
[bp
->b_y
+ 1][bp
->b_x
])) {
395 if (isplayer(Maze
[bp
->b_y
][bp
->b_x
+ 1])) {
401 * Find out what directions are clear
404 if (!iswall(bp
->b_y
, bp
->b_x
- 1))
405 mask
|= WEST
, count
++;
406 if (!iswall(bp
->b_y
- 1, bp
->b_x
))
407 mask
|= NORTH
, count
++;
408 if (!iswall(bp
->b_y
+ 1, bp
->b_x
))
409 mask
|= SOUTH
, count
++;
410 if (!iswall(bp
->b_y
, bp
->b_x
+ 1))
411 mask
|= EAST
, count
++;
414 * All blocked up, just you wait
420 * Only one way to go.
428 * Get rid of the direction that we came from
430 switch (bp
->b_face
) {
433 mask
&= ~EAST
, count
--;
437 mask
&= ~WEST
, count
--;
441 mask
&= ~SOUTH
, count
--;
445 mask
&= ~NORTH
, count
--;
450 * Pick one of the remaining directions
453 if (n
>= 0 && mask
& NORTH
)
455 if (n
>= 0 && mask
& SOUTH
)
457 if (n
>= 0 && mask
& EAST
)
459 if (n
>= 0 && mask
& WEST
)
463 * Now that we know the direction of movement,
464 * just update the position of the drone
485 switch (Maze
[bp
->b_y
][bp
->b_x
]) {
491 * give the person a chance to catch a
492 * drone if s/he is facing it
494 if (rand_num(100) < 1 &&
495 opposite(bp
->b_face
, Maze
[bp
->b_y
][bp
->b_x
])) {
496 pp
= play_at(bp
->b_y
, bp
->b_x
);
497 pp
->p_ammo
+= bp
->b_charge
;
498 message(pp
, "**** Absorbed drone ****");
500 (void) snprintf(Buf
, sizeof(buf
), "%3d", pp
->p_ammo
);
501 cgoto(pp
, STAT_AMMO_ROW
, STAT_VALUE_COL
);
514 * Put this bullet back onto the bullet list
517 save_bullet(BULLET
*bp
)
519 bp
->b_over
= Maze
[bp
->b_y
][bp
->b_x
];
520 switch (bp
->b_over
) {
534 find_under(Bullets
, bp
);
538 switch (bp
->b_over
) {
556 Maze
[bp
->b_y
][bp
->b_x
] = bp
->b_type
;
560 bp
->b_next
= Bullets
;
566 * Update the position of a player in flight
569 move_flyer(PLAYER
*pp
)
573 if (pp
->p_undershot
) {
574 fixshots(pp
->p_y
, pp
->p_x
, pp
->p_over
);
575 pp
->p_undershot
= false;
577 Maze
[pp
->p_y
][pp
->p_x
] = pp
->p_over
;
578 x
= pp
->p_x
+ pp
->p_flyx
;
579 y
= pp
->p_y
+ pp
->p_flyy
;
582 pp
->p_flyx
= -pp
->p_flyx
;
584 else if (x
> WIDTH
- 2) {
585 x
= (WIDTH
- 2) - (x
- (WIDTH
- 2));
586 pp
->p_flyx
= -pp
->p_flyx
;
590 pp
->p_flyy
= -pp
->p_flyy
;
592 else if (y
> HEIGHT
- 2) {
593 y
= (HEIGHT
- 2) - (y
- (HEIGHT
- 2));
594 pp
->p_flyy
= -pp
->p_flyy
;
597 switch (Maze
[y
][x
]) {
599 switch (rand_num(4)) {
601 PLUS_DELTA(x
, WIDTH
- 2);
607 PLUS_DELTA(y
, HEIGHT
- 2);
624 if (pp
->p_flying
== 0)
632 if (pp
->p_flying
-- == 0) {
634 if (pp
->p_face
!= BOOT
&& pp
->p_face
!= BOOT_PAIR
) {
636 checkdam(pp
, NULL
, NULL
,
637 rand_num(pp
->p_damage
/ 5), FALL
);
638 pp
->p_face
= rand_dir();
643 if (Maze
[y
][x
] == BOOT
)
644 pp
->p_face
= BOOT_PAIR
;
649 pp
->p_over
= Maze
[y
][x
];
650 Maze
[y
][x
] = pp
->p_face
;
651 showexpl(y
, x
, pp
->p_face
);
659 chkshot(BULLET
*bp
, BULLET
*next
)
668 switch (bp
->b_type
) {
675 delta
= bp
->b_size
- 1;
692 for (y
= bp
->b_y
- delta
; y
<= bp
->b_y
+ delta
; y
++) {
693 if (y
< 0 || y
>= HEIGHT
)
696 absdy
= (dy
< 0) ? -dy
: dy
;
697 for (x
= bp
->b_x
- delta
; x
<= bp
->b_x
+ delta
; x
++) {
698 if (x
< 0 || x
>= WIDTH
)
702 expl
= (dy
== 0) ? '*' : '|';
711 showexpl(y
, x
, expl
);
712 switch (Maze
[y
][x
]) {
723 damage
= bp
->b_size
- absdy
;
725 damage
= bp
->b_size
- dx
;
727 checkdam(pp
, bp
->b_owner
, bp
->b_score
,
728 damage
* MINDAM
, bp
->b_type
);
732 add_shot((Maze
[y
][x
] == GMINE
) ?
735 (Maze
[y
][x
] == GMINE
) ?
748 * handle slime shot exploding
751 chkslime(BULLET
*bp
, BULLET
*next
)
755 switch (Maze
[bp
->b_y
][bp
->b_x
]) {
766 switch (bp
->b_face
) {
782 nbp
= malloc(sizeof(*nbp
));
785 move_slime(nbp
, nbp
->b_type
== SLIME
? SLIMESPEED
: LAVASPEED
, next
);
787 move_slime(nbp
, SLIMESPEED
, next
);
793 * move the given slime shot speed times and add it back if
794 * it hasn't fizzled yet
797 move_slime(BULLET
*bp
, int speed
, BULLET
*next
)
799 int i
, j
, dirmask
, count
;
804 if (bp
->b_charge
<= 0)
812 showexpl(bp
->b_y
, bp
->b_x
, bp
->b_type
== LAVA
? LAVA
: '*');
814 showexpl(bp
->b_y
, bp
->b_x
, '*');
816 switch (Maze
[bp
->b_y
][bp
->b_x
]) {
824 pp
= play_at(bp
->b_y
, bp
->b_x
);
825 message(pp
, "You've been slimed.");
826 checkdam(pp
, bp
->b_owner
, bp
->b_score
, MINDAM
, bp
->b_type
);
835 explshot(next
, bp
->b_y
, bp
->b_x
);
836 explshot(Bullets
, bp
->b_y
, bp
->b_x
);
840 if (--bp
->b_charge
<= 0) {
847 switch (bp
->b_face
) {
849 if (!iswall(bp
->b_y
, bp
->b_x
- 1))
850 dirmask
|= WEST
, count
++;
851 if (!iswall(bp
->b_y
- 1, bp
->b_x
))
852 dirmask
|= NORTH
, count
++;
853 if (!iswall(bp
->b_y
+ 1, bp
->b_x
))
854 dirmask
|= SOUTH
, count
++;
856 if (!iswall(bp
->b_y
, bp
->b_x
+ 1))
857 dirmask
|= EAST
, count
++;
860 if (!iswall(bp
->b_y
, bp
->b_x
+ 1))
861 dirmask
|= EAST
, count
++;
862 if (!iswall(bp
->b_y
- 1, bp
->b_x
))
863 dirmask
|= NORTH
, count
++;
864 if (!iswall(bp
->b_y
+ 1, bp
->b_x
))
865 dirmask
|= SOUTH
, count
++;
867 if (!iswall(bp
->b_y
, bp
->b_x
- 1))
868 dirmask
|= WEST
, count
++;
871 if (!iswall(bp
->b_y
- 1, bp
->b_x
))
872 dirmask
|= NORTH
, count
++;
873 if (!iswall(bp
->b_y
, bp
->b_x
- 1))
874 dirmask
|= WEST
, count
++;
875 if (!iswall(bp
->b_y
, bp
->b_x
+ 1))
876 dirmask
|= EAST
, count
++;
878 if (!iswall(bp
->b_y
+ 1, bp
->b_x
))
879 dirmask
|= SOUTH
, count
++;
882 if (!iswall(bp
->b_y
+ 1, bp
->b_x
))
883 dirmask
|= SOUTH
, count
++;
884 if (!iswall(bp
->b_y
, bp
->b_x
- 1))
885 dirmask
|= WEST
, count
++;
886 if (!iswall(bp
->b_y
, bp
->b_x
+ 1))
887 dirmask
|= EAST
, count
++;
889 if (!iswall(bp
->b_y
- 1, bp
->b_x
))
890 dirmask
|= NORTH
, count
++;
895 * No place to go. Just sit here for a while and wait
896 * for adjacent squares to clear out.
901 if (bp
->b_charge
< count
) {
902 /* Only bp->b_charge paths may be taken */
903 while (count
> bp
->b_charge
) {
906 else if (dirmask
& EAST
)
908 else if (dirmask
& NORTH
)
910 else if (dirmask
& SOUTH
)
916 i
= bp
->b_charge
/ count
;
917 j
= bp
->b_charge
% count
;
918 if (dirmask
& WEST
) {
920 nbp
= create_shot(bp
->b_type
, bp
->b_y
, bp
->b_x
- 1, LEFTS
,
921 i
, bp
->b_size
, bp
->b_owner
, bp
->b_score
, true, SPACE
);
922 move_slime(nbp
, speed
- 1, next
);
924 if (dirmask
& EAST
) {
926 nbp
= create_shot(bp
->b_type
, bp
->b_y
, bp
->b_x
+ 1, RIGHT
,
927 (count
< j
) ? i
+ 1 : i
, bp
->b_size
, bp
->b_owner
,
928 bp
->b_score
, true, SPACE
);
929 move_slime(nbp
, speed
- 1, next
);
931 if (dirmask
& NORTH
) {
933 nbp
= create_shot(bp
->b_type
, bp
->b_y
- 1, bp
->b_x
, ABOVE
,
934 (count
< j
) ? i
+ 1 : i
, bp
->b_size
, bp
->b_owner
,
935 bp
->b_score
, true, SPACE
);
936 move_slime(nbp
, speed
- 1, next
);
938 if (dirmask
& SOUTH
) {
940 nbp
= create_shot(bp
->b_type
, bp
->b_y
+ 1, bp
->b_x
, BELOW
,
941 (count
< j
) ? i
+ 1 : i
, bp
->b_size
, bp
->b_owner
,
942 bp
->b_score
, true, SPACE
);
943 move_slime(nbp
, speed
- 1, next
);
951 * returns whether the given location is a wall
956 if (y
< 0 || x
< 0 || y
>= HEIGHT
|| x
>= WIDTH
)
958 switch (Maze
[y
][x
]) {
983 * Take a shot out of the air.
986 zapshot(BULLET
*blist
, BULLET
*obp
)
992 for (bp
= blist
; bp
!= NULL
; bp
= bp
->b_next
) {
993 if (bp
->b_x
!= obp
->b_x
|| bp
->b_y
!= obp
->b_y
)
995 if (bp
->b_face
== obp
->b_face
)
1002 explshot(blist
, obp
->b_y
, obp
->b_x
);
1007 * Make all shots at this location blow up
1010 explshot(BULLET
*blist
, int y
, int x
)
1014 for (bp
= blist
; bp
!= NULL
; bp
= bp
->b_next
)
1015 if (bp
->b_x
== x
&& bp
->b_y
== y
) {
1017 if (bp
->b_owner
!= NULL
)
1018 message(bp
->b_owner
, "Shot intercepted");
1024 * Return a pointer to the player at the given location
1027 play_at(int y
, int x
)
1031 for (pp
= Player
; pp
< End_player
; pp
++)
1032 if (pp
->p_x
== x
&& pp
->p_y
== y
)
1034 errx(1, "driver: couldn't find player at (%d,%d)", x
, y
);
1040 * Return true if the bullet direction faces the opposite direction
1041 * of the player in the maze
1044 opposite(int face
, char dir
)
1048 return (dir
== RIGHT
);
1050 return (dir
== LEFTS
);
1052 return (dir
== BELOW
);
1054 return (dir
== ABOVE
);
1062 * Is there a bullet at the given coordinates? If so, return
1063 * a pointer to the bullet, otherwise return NULL
1066 is_bullet(int y
, int x
)
1070 for (bp
= Bullets
; bp
!= NULL
; bp
= bp
->b_next
)
1071 if (bp
->b_y
== y
&& bp
->b_x
== x
)
1078 * change the underlying character of the shots at a location
1079 * to the given character.
1082 fixshots(int y
, int x
, char over
)
1086 for (bp
= Bullets
; bp
!= NULL
; bp
= bp
->b_next
)
1087 if (bp
->b_y
== y
&& bp
->b_x
== x
)
1093 * find the underlying character for a bullet when it lands
1094 * on another bullet.
1097 find_under(BULLET
*blist
, BULLET
*bp
)
1101 for (nbp
= blist
; nbp
!= NULL
; nbp
= nbp
->b_next
)
1102 if (bp
->b_y
== nbp
->b_y
&& bp
->b_x
== nbp
->b_x
) {
1103 bp
->b_over
= nbp
->b_over
;
1110 * mark a player as under a shot
1113 mark_player(BULLET
*bp
)
1117 for (pp
= Player
; pp
< End_player
; pp
++)
1118 if (pp
->p_y
== bp
->b_y
&& pp
->p_x
== bp
->b_x
) {
1119 pp
->p_undershot
= true;
1127 * mark a boot as under a shot
1130 mark_boot(BULLET
*bp
)
1134 for (pp
= Boot
; pp
< &Boot
[NBOOTS
]; pp
++)
1135 if (pp
->p_y
== bp
->b_y
&& pp
->p_x
== bp
->b_x
) {
1136 pp
->p_undershot
= true;