• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2 Copyright (C) 1996-1997 Id Software, Inc.
3 
4 This program is free software; you can redistribute it and/or
5 modify it under the terms of the GNU General Public License
6 as published by the Free Software Foundation; either version 2
7 of the License, or (at your option) any later version.
8 
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12 
13 See the GNU General Public License for more details.
14 
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
18 
19 */
20 // sv_move.c -- monster movement
21 
22 #include "qwsvdef.h"
23 
24 #define	STEPSIZE	18
25 
26 /*
27 =============
28 SV_CheckBottom
29 
30 Returns false if any part of the bottom of the entity is off an edge that
31 is not a staircase.
32 
33 =============
34 */
35 int c_yes, c_no;
36 
SV_CheckBottom(edict_t * ent)37 qboolean SV_CheckBottom (edict_t *ent)
38 {
39 	vec3_t	mins, maxs, start, stop;
40 	trace_t	trace;
41 	int		x, y;
42 	float	mid, bottom;
43 
44 	VectorAdd (ent->v.origin, ent->v.mins, mins);
45 	VectorAdd (ent->v.origin, ent->v.maxs, maxs);
46 
47 // if all of the points under the corners are solid world, don't bother
48 // with the tougher checks
49 // the corners must be within 16 of the midpoint
50 	start[2] = mins[2] - 1;
51 	for	(x=0 ; x<=1 ; x++)
52 		for	(y=0 ; y<=1 ; y++)
53 		{
54 			start[0] = x ? maxs[0] : mins[0];
55 			start[1] = y ? maxs[1] : mins[1];
56 			if (SV_PointContents (start) != CONTENTS_SOLID)
57 				goto realcheck;
58 		}
59 
60 	c_yes++;
61 	return true;		// we got out easy
62 
63 realcheck:
64 	c_no++;
65 //
66 // check it for real...
67 //
68 	start[2] = mins[2];
69 
70 // the midpoint must be within 16 of the bottom
71 	start[0] = stop[0] = (mins[0] + maxs[0])*0.5;
72 	start[1] = stop[1] = (mins[1] + maxs[1])*0.5;
73 	stop[2] = start[2] - 2*STEPSIZE;
74 	trace = SV_Move (start, vec3_origin, vec3_origin, stop, true, ent);
75 
76 	if (trace.fraction == 1.0)
77 		return false;
78 	mid = bottom = trace.endpos[2];
79 
80 // the corners must be within 16 of the midpoint
81 	for	(x=0 ; x<=1 ; x++)
82 		for	(y=0 ; y<=1 ; y++)
83 		{
84 			start[0] = stop[0] = x ? maxs[0] : mins[0];
85 			start[1] = stop[1] = y ? maxs[1] : mins[1];
86 
87 			trace = SV_Move (start, vec3_origin, vec3_origin, stop, true, ent);
88 
89 			if (trace.fraction != 1.0 && trace.endpos[2] > bottom)
90 				bottom = trace.endpos[2];
91 			if (trace.fraction == 1.0 || mid - trace.endpos[2] > STEPSIZE)
92 				return false;
93 		}
94 
95 	c_yes++;
96 	return true;
97 }
98 
99 
100 /*
101 =============
102 SV_movestep
103 
104 Called by monster program code.
105 The move will be adjusted for slopes and stairs, but if the move isn't
106 possible, no move is done, false is returned, and
107 pr_global_struct->trace_normal is set to the normal of the blocking wall
108 =============
109 */
SV_movestep(edict_t * ent,vec3_t move,qboolean relink)110 qboolean SV_movestep (edict_t *ent, vec3_t move, qboolean relink)
111 {
112 	float		dz;
113 	vec3_t		oldorg, neworg, end;
114 	trace_t		trace;
115 	int			i;
116 	edict_t		*enemy;
117 
118 // try the move
119 	VectorCopy (ent->v.origin, oldorg);
120 	VectorAdd (ent->v.origin, move, neworg);
121 
122 // flying monsters don't step up
123 	if ( (int)ent->v.flags & (FL_SWIM | FL_FLY) )
124 	{
125 	// try one move with vertical motion, then one without
126 		for (i=0 ; i<2 ; i++)
127 		{
128 			VectorAdd (ent->v.origin, move, neworg);
129 			enemy = PROG_TO_EDICT(ent->v.enemy);
130 			if (i == 0 && enemy != sv.edicts)
131 			{
132 				dz = ent->v.origin[2] - PROG_TO_EDICT(ent->v.enemy)->v.origin[2];
133 				if (dz > 40)
134 					neworg[2] -= 8;
135 				if (dz < 30)
136 					neworg[2] += 8;
137 			}
138 			trace = SV_Move (ent->v.origin, ent->v.mins, ent->v.maxs, neworg, false, ent);
139 
140 			if (trace.fraction == 1)
141 			{
142 				if ( ((int)ent->v.flags & FL_SWIM) && SV_PointContents(trace.endpos) == CONTENTS_EMPTY )
143 					return false;	// swim monster left water
144 
145 				VectorCopy (trace.endpos, ent->v.origin);
146 				if (relink)
147 					SV_LinkEdict (ent, true);
148 				return true;
149 			}
150 
151 			if (enemy == sv.edicts)
152 				break;
153 		}
154 
155 		return false;
156 	}
157 
158 // push down from a step height above the wished position
159 	neworg[2] += STEPSIZE;
160 	VectorCopy (neworg, end);
161 	end[2] -= STEPSIZE*2;
162 
163 	trace = SV_Move (neworg, ent->v.mins, ent->v.maxs, end, false, ent);
164 
165 	if (trace.allsolid)
166 		return false;
167 
168 	if (trace.startsolid)
169 	{
170 		neworg[2] -= STEPSIZE;
171 		trace = SV_Move (neworg, ent->v.mins, ent->v.maxs, end, false, ent);
172 		if (trace.allsolid || trace.startsolid)
173 			return false;
174 	}
175 	if (trace.fraction == 1)
176 	{
177 	// if monster had the ground pulled out, go ahead and fall
178 		if ( (int)ent->v.flags & FL_PARTIALGROUND )
179 		{
180 			VectorAdd (ent->v.origin, move, ent->v.origin);
181 			if (relink)
182 				SV_LinkEdict (ent, true);
183 			ent->v.flags = (int)ent->v.flags & ~FL_ONGROUND;
184 //	Con_Printf ("fall down\n");
185 			return true;
186 		}
187 
188 		return false;		// walked off an edge
189 	}
190 
191 // check point traces down for dangling corners
192 	VectorCopy (trace.endpos, ent->v.origin);
193 
194 	if (!SV_CheckBottom (ent))
195 	{
196 		if ( (int)ent->v.flags & FL_PARTIALGROUND )
197 		{	// entity had floor mostly pulled out from underneath it
198 			// and is trying to correct
199 			if (relink)
200 				SV_LinkEdict (ent, true);
201 			return true;
202 		}
203 		VectorCopy (oldorg, ent->v.origin);
204 		return false;
205 	}
206 
207 	if ( (int)ent->v.flags & FL_PARTIALGROUND )
208 	{
209 //		Con_Printf ("back on ground\n");
210 		ent->v.flags = (int)ent->v.flags & ~FL_PARTIALGROUND;
211 	}
212 	ent->v.groundentity = EDICT_TO_PROG(trace.ent);
213 
214 // the move is ok
215 	if (relink)
216 		SV_LinkEdict (ent, true);
217 	return true;
218 }
219 
220 
221 //============================================================================
222 
223 /*
224 ======================
225 SV_StepDirection
226 
227 Turns to the movement direction, and walks the current distance if
228 facing it.
229 
230 ======================
231 */
232 void PF_changeyaw (void);
SV_StepDirection(edict_t * ent,float yaw,float dist)233 qboolean SV_StepDirection (edict_t *ent, float yaw, float dist)
234 {
235 	vec3_t		move, oldorigin;
236 	float		delta;
237 
238 	ent->v.ideal_yaw = yaw;
239 	PF_changeyaw();
240 
241 	yaw = yaw*M_PI*2 / 360;
242 	move[0] = cos(yaw)*dist;
243 	move[1] = sin(yaw)*dist;
244 	move[2] = 0;
245 
246 	VectorCopy (ent->v.origin, oldorigin);
247 	if (SV_movestep (ent, move, false))
248 	{
249 		delta = ent->v.angles[YAW] - ent->v.ideal_yaw;
250 		if (delta > 45 && delta < 315)
251 		{		// not turned far enough, so don't take the step
252 			VectorCopy (oldorigin, ent->v.origin);
253 		}
254 		SV_LinkEdict (ent, true);
255 		return true;
256 	}
257 	SV_LinkEdict (ent, true);
258 
259 	return false;
260 }
261 
262 /*
263 ======================
264 SV_FixCheckBottom
265 
266 ======================
267 */
SV_FixCheckBottom(edict_t * ent)268 void SV_FixCheckBottom (edict_t *ent)
269 {
270 //	Con_Printf ("SV_FixCheckBottom\n");
271 
272 	ent->v.flags = (int)ent->v.flags | FL_PARTIALGROUND;
273 }
274 
275 
276 
277 /*
278 ================
279 SV_NewChaseDir
280 
281 ================
282 */
283 #define	DI_NODIR	-1
SV_NewChaseDir(edict_t * actor,edict_t * enemy,float dist)284 void SV_NewChaseDir (edict_t *actor, edict_t *enemy, float dist)
285 {
286 	float		deltax,deltay;
287 	float			d[3];
288 	float		tdir, olddir, turnaround;
289 
290 	olddir = anglemod( (int)(actor->v.ideal_yaw/45)*45 );
291 	turnaround = anglemod(olddir - 180);
292 
293 	deltax = enemy->v.origin[0] - actor->v.origin[0];
294 	deltay = enemy->v.origin[1] - actor->v.origin[1];
295 	if (deltax>10)
296 		d[1]= 0;
297 	else if (deltax<-10)
298 		d[1]= 180;
299 	else
300 		d[1]= DI_NODIR;
301 	if (deltay<-10)
302 		d[2]= 270;
303 	else if (deltay>10)
304 		d[2]= 90;
305 	else
306 		d[2]= DI_NODIR;
307 
308 // try direct route
309 	if (d[1] != DI_NODIR && d[2] != DI_NODIR)
310 	{
311 		if (d[1] == 0)
312 			tdir = d[2] == 90 ? 45 : 315;
313 		else
314 			tdir = d[2] == 90 ? 135 : 215;
315 
316 		if (tdir != turnaround && SV_StepDirection(actor, tdir, dist))
317 			return;
318 	}
319 
320 // try other directions
321 	if ( ((rand()&3) & 1) ||  abs(deltay)>abs(deltax))
322 	{
323 		tdir=d[1];
324 		d[1]=d[2];
325 		d[2]=tdir;
326 	}
327 
328 	if (d[1]!=DI_NODIR && d[1]!=turnaround
329 	&& SV_StepDirection(actor, d[1], dist))
330 			return;
331 
332 	if (d[2]!=DI_NODIR && d[2]!=turnaround
333 	&& SV_StepDirection(actor, d[2], dist))
334 			return;
335 
336 /* there is no direct path to the player, so pick another direction */
337 
338 	if (olddir!=DI_NODIR && SV_StepDirection(actor, olddir, dist))
339 			return;
340 
341 	if (rand()&1) 	/*randomly determine direction of search*/
342 	{
343 		for (tdir=0 ; tdir<=315 ; tdir += 45)
344 			if (tdir!=turnaround && SV_StepDirection(actor, tdir, dist) )
345 					return;
346 	}
347 	else
348 	{
349 		for (tdir=315 ; tdir >=0 ; tdir -= 45)
350 			if (tdir!=turnaround && SV_StepDirection(actor, tdir, dist) )
351 					return;
352 	}
353 
354 	if (turnaround != DI_NODIR && SV_StepDirection(actor, turnaround, dist) )
355 			return;
356 
357 	actor->v.ideal_yaw = olddir;		// can't move
358 
359 // if a bridge was pulled out from underneath a monster, it may not have
360 // a valid standing position at all
361 
362 	if (!SV_CheckBottom (actor))
363 		SV_FixCheckBottom (actor);
364 
365 }
366 
367 /*
368 ======================
369 SV_CloseEnough
370 
371 ======================
372 */
SV_CloseEnough(edict_t * ent,edict_t * goal,float dist)373 qboolean SV_CloseEnough (edict_t *ent, edict_t *goal, float dist)
374 {
375 	int		i;
376 
377 	for (i=0 ; i<3 ; i++)
378 	{
379 		if (goal->v.absmin[i] > ent->v.absmax[i] + dist)
380 			return false;
381 		if (goal->v.absmax[i] < ent->v.absmin[i] - dist)
382 			return false;
383 	}
384 	return true;
385 }
386 
387 /*
388 ======================
389 SV_MoveToGoal
390 
391 ======================
392 */
SV_MoveToGoal(void)393 void SV_MoveToGoal (void)
394 {
395 	edict_t		*ent, *goal;
396 	float		dist;
397 
398 	ent = PROG_TO_EDICT(pr_global_struct->self);
399 	goal = PROG_TO_EDICT(ent->v.goalentity);
400 	dist = G_FLOAT(OFS_PARM0);
401 
402 	if ( !( (int)ent->v.flags & (FL_ONGROUND|FL_FLY|FL_SWIM) ) )
403 	{
404 		G_FLOAT(OFS_RETURN) = 0;
405 		return;
406 	}
407 
408 // if the next step hits the enemy, return immediately
409 	if ( PROG_TO_EDICT(ent->v.enemy) != sv.edicts &&  SV_CloseEnough (ent, goal, dist) )
410 		return;
411 
412 // bump around...
413 	if ( (rand()&3)==1 ||
414 	!SV_StepDirection (ent, ent->v.ideal_yaw, dist))
415 	{
416 		SV_NewChaseDir (ent, goal, dist);
417 	}
418 }
419 
420