CnC_Remastered_Collection

Command and Conquer: Red Alert
Log | Files | Refs | README | LICENSE

FOOT.CPP (107668B)


      1 //
      2 // Copyright 2020 Electronic Arts Inc.
      3 //
      4 // TiberianDawn.DLL and RedAlert.dll and corresponding source code is free 
      5 // software: you can redistribute it and/or modify it under the terms of 
      6 // the GNU General Public License as published by the Free Software Foundation, 
      7 // either version 3 of the License, or (at your option) any later version.
      8 
      9 // TiberianDawn.DLL and RedAlert.dll and corresponding source code is distributed 
     10 // in the hope that it will be useful, but with permitted additional restrictions 
     11 // under Section 7 of the GPL. See the GNU General Public License in LICENSE.TXT 
     12 // distributed with this program. You should have received a copy of the 
     13 // GNU General Public License along with permitted additional restrictions 
     14 // with this program. If not, see https://github.com/electronicarts/CnC_Remastered_Collection
     15 
     16 /* $Header:   F:\projects\c&c\vcs\code\foot.cpv   2.17   16 Oct 1995 16:51:42   JOE_BOSTIC  $ */
     17 /***********************************************************************************************
     18  ***             C O N F I D E N T I A L  ---  W E S T W O O D   S T U D I O S               ***
     19  ***********************************************************************************************
     20  *                                                                                             *
     21  *                 Project Name : Command & Conquer                                            *
     22  *                                                                                             *
     23  *                    File Name : FOOT.CPP                                                     *
     24  *                                                                                             *
     25  *                   Programmer : Joe L. Bostic                                                *
     26  *                                                                                             *
     27  *                   Start Date : April 22, 1994                                               *
     28  *                                                                                             *
     29  *                  Last Update : August 13, 1995 [JLB]                                        *
     30  *                                                                                             *
     31  *---------------------------------------------------------------------------------------------*
     32  * Functions:                                                                                  *
     33  *   FootClass::Active_Click_With -- Intiates attack or move according to target clicked on.   *
     34  *   FootClass::Active_Click_With -- Performs action as a result of left mouse click.          *
     35  *   FootClass::Approach_Target -- Sets the navigation computer to approach target object.     *
     36  *   FootClass::Assign_Destination -- Assigns specified destination to NavCom.                 *
     37  *   FootClass::Assign_Mission -- Assign mission to foot class object.                         *
     38  *   FootClass::Basic_Path -- Finds the basic path for a ground object.                        *
     39  *   FootClass::Body_Facing -- Set the body rotation/facing.                                   *
     40  *   FootClass::Can_Demolish -- Checks to see if this object can be sold back.                 *
     41  *   FootClass::Can_Enter_Cell -- Checks to see if the object can enter cell specified.        *
     42  *   FootClass::Death_Announcement -- Announces the death of a unit.                           *
     43  *   FootClass::Debug_Dump -- Displays the status of the FootClass to the mono monitor.        *
     44  *   FootClass::Detach -- Detaches a target from tracking systems.                             *
     45  *   FootClass::Detach_All -- Removes this object from the game system.                        *
     46  *   FootClass::Enters_Building -- When unit enters a building for some reason.                *
     47  *   FootClass::FootClass -- Default constructor for foot class objects.                       *
     48  *   FootClass::FootClass -- Normal constructor for the foot class object.                     *
     49  *   FootClass::Greatest_Threat -- Fetches the greatest threat to this object.                 *
     50  *   FootClass::Likely_Coord -- Fetches the coordinate the object will be at shortly.          *
     51  *   FootClass::Limbo -- Intercepts limbo event and handles FootClass processing.              *
     52  *   FootClass::Mark -- Unit interface to map rendering system.                                *
     53  *   FootClass::Mission_Attack -- AI for heading towards and firing upon target.               *
     54  *   FootClass::Mission_Capture -- Handles the capture mission.                                *
     55  *   FootClass::Mission_Enter -- Enter (cooperatively) mission handler.                        *
     56  *   FootClass::Mission_Guard_Area -- Causes unit to guard an area about twice weapon range.   *
     57  *   FootClass::Mission_Hunt -- Handles the default hunt order.                                *
     58  *   FootClass::Mission_Move -- AI process for moving a vehicle to its destination.            *
     59  *   FootClass::Offload_Tiberium_Bail -- Fetches the Tiberium to offload per step.             *
     60  *   FootClass::Override_Mission -- temporarily overides a units mission                       *
     61  *   FootClass::Per_Cell_Process -- Perform action based on once-per-cell condition.           *
     62  *   FootClass::Receive_Message -- Movement related radio messages are handled here.           *
     63  *   FootClass::Rescue_Mission -- Calls this unit to the rescue.                               *
     64  *   FootClass::Restore_Mission -- Restores an overidden mission                               *
     65  *   FootClass::Sell_Back -- Causes this object to be sold back.                               *
     66  *   FootClass::Set_Speed -- Initiate unit movement physics.                                   *
     67  *   FootClass::Sort_Y -- Determine the sort coordinate for foot class objects.                *
     68  *   FootClass::Start_Driver -- This starts the driver heading to the destination desired.     *
     69  *   FootClass::Stop_Driver -- This routine clears the driving state of the object.            *
     70  *   FootClass::Stun -- Prepares a ground travelling object for removal.                       *
     71  *   FootClass::Take_Damage -- Handles taking damage to this object.                           *
     72  *   FootClass::Unlimbo -- Unlimbos object and performs special fixups.                        *
     73  *   FootClass::~FootClass -- Default destructor for foot class objects.                       *
     74  * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     75 
     76 #include	"function.h"
     77 
     78 
     79 /***********************************************************************************************
     80  * FootClass::FootClass -- Default constructor for foot class objects.                         *
     81  *                                                                                             *
     82  *    This is the default constructor for FootClass objects. It sets the foot class values to  *
     83  *    their default starting settings.                                                         *
     84  *                                                                                             *
     85  * INPUT:   none                                                                               *
     86  *                                                                                             *
     87  * OUTPUT:  none                                                                               *
     88  *                                                                                             *
     89  * WARNINGS:   none                                                                            *
     90  *                                                                                             *
     91  * HISTORY:                                                                                    *
     92  *   11/23/1994 JLB : Created.                                                                 *
     93  *=============================================================================================*/
     94 FootClass::FootClass(void) : Speed(0)
     95 {
     96 	IsDriving = false;
     97 	IsInitiated = false;
     98 	IsPlanningToLook = false;
     99 	IsDeploying = false;
    100 	IsNewNavCom = false;
    101 	IsFiring = false;
    102 	IsRotating = false;
    103 	IsUnloading = false;
    104 	NavCom = TARGET_NONE;
    105 	SuspendedNavCom = TARGET_NONE;
    106 	Path[0] = FACING_NONE;
    107 	HeadToCoord = NULL;
    108 	Member = 0;
    109 	Team = 0;
    110 	PathDelay = 0;
    111 	TryTryAgain = PATH_RETRY;
    112 	if (House) {
    113 		House->CurUnits++;
    114 	}
    115 	Group = -1;
    116 }
    117 
    118 
    119 /***********************************************************************************************
    120  * FootClass::~FootClass -- Default destructor for foot class objects.                         *
    121  *                                                                                             *
    122  *    At this level of the destruction process, the house record of the number of units        *
    123  *    currently in inventory is decremented to reflect this units destruction.                 *
    124  *                                                                                             *
    125  * INPUT:   none                                                                               *
    126  *                                                                                             *
    127  * OUTPUT:  none                                                                               *
    128  *                                                                                             *
    129  * WARNINGS:   none                                                                            *
    130  *                                                                                             *
    131  * HISTORY:                                                                                    *
    132  *   01/10/1995 JLB : Created.                                                                 *
    133  *=============================================================================================*/
    134 FootClass::~FootClass(void)
    135 {
    136 	if (GameActive && House) {
    137 		House->CurUnits--;
    138 	}
    139 }
    140 
    141 
    142 /***********************************************************************************************
    143  * FootClass::FootClass -- Normal constructor for the foot class object.                       *
    144  *                                                                                             *
    145  *    This is the normal constructor used when creating a foot class object.                   *
    146  *                                                                                             *
    147  * INPUT:   house    -- The house that owns this object.                                       *
    148  *                                                                                             *
    149  * OUTPUT:  none                                                                               *
    150  *                                                                                             *
    151  * WARNINGS:   none                                                                            *
    152  *                                                                                             *
    153  * HISTORY:                                                                                    *
    154  *   12/29/1994 JLB : Created.                                                                 *
    155  *=============================================================================================*/
    156 FootClass::FootClass(HousesType house) :
    157 			TechnoClass(house),
    158 			Speed(0)
    159 {
    160 	Member = 0;
    161 	Team = 0;
    162 	Path[0] = FACING_NONE;
    163 	NavCom = TARGET_NONE;
    164 	SuspendedNavCom = TARGET_NONE;
    165 	IsUnloading = false;
    166 	IsDriving = false;
    167 	IsInitiated = false;
    168 	IsRotating = false;
    169 	IsFiring = false;
    170 	IsDeploying = false;
    171 	IsNewNavCom = false;
    172 	IsPlanningToLook = false;
    173 	HeadToCoord = 0L;
    174 	PathDelay = 0;
    175 	Group = -1;
    176 	TryTryAgain = PATH_RETRY;
    177 	House->CurUnits++;
    178 }
    179 
    180 
    181 #ifdef CHEAT_KEYS
    182 /***********************************************************************************************
    183  * FootClass::Debug_Dump -- Displays the status of the FootClass to the mono monitor.          *
    184  *                                                                                             *
    185  *    This routine is used to output the current status of the foot class to the mono          *
    186  *    monitor. Through this display bugs may be tracked down or eliminated.                    *
    187  *                                                                                             *
    188  * INPUT:   none                                                                               *
    189  *                                                                                             *
    190  * OUTPUT:  none                                                                               *
    191  *                                                                                             *
    192  * WARNINGS:   none                                                                            *
    193  *                                                                                             *
    194  * HISTORY:                                                                                    *
    195  *   06/02/1994 JLB : Created.                                                                 *
    196  *   07/04/1995 JLB : Handles aircraft special case.                                           *
    197  *=============================================================================================*/
    198 void FootClass::Debug_Dump(MonoClass *mono) const
    199 {
    200 	static char	const * _p2c[9] = {"-","0","1","2","3","4","5","6","7"};
    201 #define	Path_To_String(a)	_p2c[((ABS((int)a+1))%9)]
    202 
    203 	/*
    204 	**	Display the common data for all objects that inherity from FootClass.
    205 	*/
    206 	mono->Set_Cursor(63, 7);
    207 	if (Team) {
    208 		mono->Printf("%s(%d)", Team->Class->IniName, Teams.ID(Team));
    209 	} else {
    210 		mono->Printf("(none)");
    211 	}
    212 	mono->Set_Cursor(73, 7);mono->Printf("%04X", ArchiveTarget);
    213 	mono->Set_Cursor(42, 1);mono->Printf("%04X", NavCom);
    214 	mono->Set_Cursor(44, 3);mono->Printf("%d", Speed);
    215 
    216 	/*
    217 	**	Although aircraft inherit from FootClass, some of the variables are not
    218 	**	used and thus should not be displayed.
    219 	*/
    220 	if (What_Am_I() != RTTI_AIRCRAFT) {
    221 		mono->Set_Cursor(50, 3);
    222 		mono->Printf("%s%s%s%s%s%s%s%s%s%s%s%s",
    223 			Path_To_String(Path[0]),
    224 			Path_To_String(Path[1]),
    225 			Path_To_String(Path[2]),
    226 			Path_To_String(Path[3]),
    227 			Path_To_String(Path[4]),
    228 			Path_To_String(Path[5]),
    229 			Path_To_String(Path[6]),
    230 			Path_To_String(Path[7]),
    231 			Path_To_String(Path[8]),
    232 			Path_To_String(Path[9]),
    233 			Path_To_String(Path[10]),
    234 			Path_To_String(Path[11]),
    235 			Path_To_String(Path[12]));
    236 
    237 		mono->Set_Cursor(65, 1);mono->Printf("%08lX", Head_To_Coord());
    238 		mono->Text_Print("X", 16 + (IsDeploying?2:0), 12);
    239 		mono->Text_Print("X", 16 + (IsRotating?2:0), 13);
    240 		mono->Text_Print("X", 16 + (IsDriving?2:0), 15);
    241 		mono->Text_Print("X", 16 + (IsFiring?2:0), 14);
    242 		mono->Text_Print("X", 16 + (IsPlanningToLook?2:0), 16);
    243 	}
    244 	TechnoClass::Debug_Dump(mono);
    245 }
    246 #endif
    247 
    248 
    249 /***********************************************************************************************
    250  * FootClass::Set_Speed -- Initiate unit movement physics.                                     *
    251  *                                                                                             *
    252  *    This routine is used to set a unit's velocity control structure.                         *
    253  *    The game will then process the unit's movement during the momentum                       *
    254  *    physics calculation.                                                                     *
    255  *                                                                                             *
    256  * INPUT:   unit  -- Pointer to the unit to alter.                                             *
    257  *                                                                                             *
    258  *          speed -- Throttle setting (0=stop, 255=full throttle).                             *
    259  *                                                                                             *
    260  * OUTPUT:  none                                                                               *
    261  *                                                                                             *
    262  * WARNINGS:   none                                                                            *
    263  *                                                                                             *
    264  * HISTORY:                                                                                    *
    265  *   09/07/1992 JLB : Created.                                                                 *
    266  *   09/24/1993 JLB : Revised for faster speed.                                                *
    267  *   04/02/1994 JLB : Revised for new system.                                                  *
    268  *   04/15/1994 JLB : Converted to member function.                                            *
    269  *   07/21/1994 JLB : Simplified.                                                              *
    270  *=============================================================================================*/
    271 void FootClass::Set_Speed(int speed)
    272 {
    273 	speed &= 0xFF;
    274 	((unsigned char &)Speed) = speed;
    275 }
    276 
    277 
    278 /***********************************************************************************************
    279  * FootClass::Mark -- Unit interface to map rendering system.                                  *
    280  *                                                                                             *
    281  *    This routine is the interface function for units as they relate to                       *
    282  *    the map rendering system. Whenever a unit's imagery changes, this                        *
    283  *    function is called.                                                                      *
    284  *                                                                                             *
    285  * INPUT:   mark  -- Type of image change (MARK_UP, _DOWN, _CHANGE)                            *
    286  *             MARK_UP  -- Unit is removed.                                                    *
    287  *             MARK_CHANGE -- Unit alters image but doesn't move.                              *
    288  *             MARK_DOWN -- Unit is overlaid onto existing icons.                              *
    289  *                                                                                             *
    290  * OUTPUT:  bool; Did the marking operation succeed? Failure could be the result of marking    *
    291  *                down when it is already down, or visa versa.                                 *
    292  *                                                                                             *
    293  * WARNINGS:   none                                                                            *
    294  *                                                                                             *
    295  * HISTORY:                                                                                    *
    296  *   09/14/1991 JLB : Created.                                                                 *
    297  *   04/15/1994 JLB : Converted to member function.                                            *
    298  *   12/23/1994 JLB : Performs low level check before processing.                              *
    299  *=============================================================================================*/
    300 bool FootClass::Mark(MarkType mark)
    301 {
    302 	if (TechnoClass::Mark(mark)) {
    303 		CELL cell = Coord_Cell(Coord);
    304 
    305 		/*
    306 		**	Inform the map of the refresh, occupation, and overlap
    307 		**	request.
    308 		**	Special case is fixed-wing aircraft, which are never placed
    309 		**	or picked up since they can never land.
    310 		*/
    311 		switch (mark) {
    312 			case MARK_UP:
    313 				if (What_Am_I() != RTTI_AIRCRAFT || !((AircraftClass*)this)->Class->IsFixedWing) {
    314 					Map.Pick_Up(cell, this);
    315 				}
    316 				break;
    317 
    318 			case MARK_DOWN:
    319 				if (What_Am_I() != RTTI_AIRCRAFT || !((AircraftClass*)this)->Class->IsFixedWing) {
    320 					Map.Place_Down(cell, this);
    321 				}
    322 				break;
    323 
    324 			default:
    325 				Map.Refresh_Cells(cell, Overlap_List());
    326 				Map.Refresh_Cells(cell, Occupy_List());
    327 				break;
    328 		}
    329 		return(true);
    330 	}
    331 	return(false);
    332 }
    333 
    334 
    335 /***********************************************************************************************
    336  * FootClass::Basic_Path -- Finds the basic path for a ground object.                          *
    337  *                                                                                             *
    338  *    This is a common routine used by both infantry and other ground travelling units. It     *
    339  *    will fill in the unit's basic path to the NavCom destination.                            *
    340  *                                                                                             *
    341  * INPUT:   none                                                                               *
    342  *                                                                                             *
    343  * OUTPUT:  bool; Was a path found? A failure to find a path means either the target cannot    *
    344  *                be found or the terrain prohibits the unit's movement.                       *
    345  *                                                                                             *
    346  * WARNINGS:   none                                                                            *
    347  *                                                                                             *
    348  * HISTORY:                                                                                    *
    349  *   10/17/1994 JLB : Created.                                                                 *
    350  *=============================================================================================*/
    351 bool FootClass::Basic_Path(void)
    352 {
    353 	PathType		*path;			// Pointer to path control structure.
    354 	CELL			cell;
    355 	int 			skip_path = false;
    356 
    357 	Path[0] = FACING_NONE;
    358 
    359 	if (Target_Legal(NavCom)) {
    360 		cell = As_Cell(NavCom);
    361 
    362 		/*
    363 		**	When the navigation computer is set to a location that is impassible, then
    364 		**	find a nearby cell that can be entered and try to head toward that instead.
    365 		**	EXCEPT when that cell is very close -- then just bail.
    366 		*/
    367 //		IsFindPath = true;
    368 		if (Can_Enter_Cell(cell) == MOVE_NO && Distance(NavCom) > 0x0300) {
    369 			static int	_faceadjust[8] = {0, 1, -1, 2, -2, 3, -3, 4};
    370 			FacingType	f2 = (FacingType)(((unsigned)::Direction(cell, Coord_Cell(Coord)))>>5);
    371 
    372 			for (unsigned index = 0; index < (sizeof(_faceadjust) / sizeof(_faceadjust[0])); index++) {
    373 				CELL	cell2;
    374 
    375 				cell2 = Adjacent_Cell(cell, (FacingType)((f2+_faceadjust[index])&0x7));
    376 				if (Can_Enter_Cell(cell2, FACING_NONE) <= MOVE_CLOAK) {
    377 					cell = cell2;
    378 					break;
    379 				}
    380 			}
    381 		}
    382 //		IsFindPath = false;
    383 
    384 #ifdef SPECIAL
    385 		if (What_Am_I() == RTTI_INFANTRY) {
    386 			CELL mycell = Coord_Cell(Center_Coord());
    387 //			Mark(MARK_UP);
    388 			ObjectClass *obj = Map[mycell].Cell_Occupier();
    389 			while (obj) {
    390 				if (obj != this && obj->What_Am_I() == RTTI_INFANTRY) {
    391 					InfantryClass *inf = (InfantryClass *)obj;
    392 					if (inf->NavCom == NavCom && inf->Path[0] != FACING_NONE) {
    393 						if (Coord_Cell(inf->Head_To_Coord()) == Coord_Cell(inf->Coord)) {
    394 							Mem_Copy(&inf->Path[1], Path, sizeof(Path)-sizeof(Path[0]));
    395 						} else {
    396 							Mem_Copy(inf->Path, Path, sizeof(Path));
    397 						}
    398 						if (Path[0] != FACING_NONE) {
    399 							skip_path = true;
    400 						}
    401 						break;
    402 					}
    403 				}
    404 				obj = obj->Next;
    405 			}
    406 //			Mark(MARK_DOWN);
    407 		}
    408 #endif
    409 
    410 		if (!skip_path) {
    411 			Mark(MARK_UP);
    412 			Path[0] = FACING_NONE;		// Probably not necessary, but...
    413 
    414 			/*
    415 			**	Try to find a path to the destination. If a failure occurs, then keep trying
    416 			**	with greater determination until either a complete failure occurs, or a decent
    417 			**	path was found.
    418 			*/
    419 			bool			found1=false;		// Found a best path yet?
    420 			PathType	   path1;
    421 			FacingType	workpath1[200];	// Staging area for path list.
    422 			FacingType	workpath2[200];	// Staging area for path list.
    423 			MoveType		maxtype = MOVE_TEMP;
    424 			if (!House->IsHuman) {
    425 				maxtype = MOVE_DESTROYABLE;
    426 			} else {
    427 
    428 				/*
    429 				**	For simple movement missions by the human player, then don't
    430 				**	consider friendly units as passable if close to the destination.
    431 				**	This will prevent a human controlled unit from just sitting next
    432 				**	to a destination just because there is another friendly unit
    433 				**	occupying the destination location.
    434 				*/
    435 				if (Mission == MISSION_MOVE && Distance(NavCom) < 0x0280) {
    436 					maxtype = MOVE_DESTROYABLE;
    437 				}
    438 			}
    439 
    440 			/*
    441 			**	Determine if ANY path could be calculated by first examining the most
    442 			**	aggressive case. If this fails, then no path will succeed. Further
    443 			**	scanning is unnecessary.
    444 			*/
    445 			path = Find_Path(cell, &workpath1[0], sizeof(workpath1), maxtype);
    446 			if (path && path->Cost) {
    447 				memcpy(&path1, path, sizeof(path1));
    448 				found1 = true;
    449 
    450 				/*
    451 				**	Scan for the best path possible. If this succeeds, then do a simple
    452 				**	comparison with the most agressive path. If they are very close, then
    453 				**	go with the best (easiest) path method.
    454 				*/
    455 				path = Find_Path(cell, &workpath2[0], sizeof(workpath2), MOVE_CLOAK);
    456 				if (path && path->Cost && path->Cost < MAX((path1.Cost + (path1.Cost/2)), 3)) {
    457 					memcpy(&path1, path, sizeof(path1));
    458 					memcpy(workpath1, workpath2, sizeof(workpath1));
    459 				} else {
    460 
    461 					/*
    462 					**	The easiest path method didn't result in a satisfactory path. Scan through
    463 					**	the rest of the path options, looking for the best one.
    464 					*/
    465 					for (MoveType move = MOVE_MOVING_BLOCK; move < maxtype; move++) {
    466 						path = Find_Path(cell, &workpath2[0], sizeof(workpath2), move);
    467 						if (path && path->Cost && path->Cost < MAX((path1.Cost + (path1.Cost/2)), 3)) {
    468 							memcpy(&path1, path, sizeof(path1));
    469 							memcpy(workpath1, workpath2, sizeof(workpath1));
    470 						}
    471 					}
    472 				}
    473 			}
    474 
    475 #ifdef OBSOLETE
    476 			for (MoveType move = MOVE_CLOAK; move <= maxtype; move++) {
    477 				if (!found1) {
    478 					path = Find_Path(cell, &workpath1[0], sizeof(workpath1), move);
    479 					if (path && path->Cost) {
    480 						memcpy(&path1, path, sizeof(path1));
    481 						found1 = true;
    482 						if (path1.Cost < 5) break;
    483 					}
    484 				} else {
    485 					path = Find_Path(cell, &workpath2[0], sizeof(workpath2), move);
    486 
    487 					if (path) {
    488 						if (path->Cost && path->Cost <= path1.Cost/2) {
    489 							memcpy(&path1, path, sizeof(path1));
    490 							memcpy(workpath1, workpath2, sizeof(workpath1));
    491 						}
    492 					}
    493 				}
    494 			}
    495 #endif
    496 
    497 			/*
    498 			**	If a good path was found, then record it in the object's path
    499 			**	list.
    500 			*/
    501 			if (found1) {
    502 				Fixup_Path(&path1);
    503 				memcpy(&Path[0], &workpath1[0], MIN(path->Length, (int)sizeof(Path)));
    504 			}
    505 
    506 			Mark(MARK_DOWN);
    507 		}
    508 
    509 
    510 #ifdef NEVER
    511 		/*
    512 		** Patch at this point to see if we are moving directly into a
    513 		** MOVE_TEMP.  This allows units to bunch up at a bridge even if there
    514 		** is an enormously long way around.  This also allows units to give
    515 		** up trying to move into the MOVE_TEMP using the normal movement
    516 		** retry logic.
    517 		*/
    518 		CELL cell = Adjacent_Cell(Coord_Cell(Coord), Path[0]);
    519 		if (Can_Enter_Cell(cell, FACING_NONE) == MOVEF_TEMP) {
    520 			Path[0] = FACING_NONE;
    521 		}
    522 #endif
    523 
    524 		PathDelay = PATH_DELAY;
    525 		if (Path[0] != FACING_NONE) return(true);
    526 
    527 		/*
    528 		**	If a basic path couldn't be determined, then abort the navigation process.
    529 		*/
    530 //		NavCom = TARGET_NONE;
    531 		Stop_Driver();
    532 	}
    533 	return(false);
    534 }
    535 
    536 
    537 /***********************************************************************************************
    538  * FootClass::Mission_Move -- AI process for moving a vehicle to its destination.              *
    539  *                                                                                             *
    540  *    This simple AI script handles moving the vehicle to its desired destination. Since       *
    541  *    simple movement is handled directly by the engine, this routine merely waits until       *
    542  *    the unit has reached its destination, and then causes the unit to enter idle mode.       *
    543  *                                                                                             *
    544  * INPUT:   none                                                                               *
    545  *                                                                                             *
    546  * OUTPUT:  Returns with the delay before calling this routine again.                          *
    547  *                                                                                             *
    548  * WARNINGS:   none                                                                            *
    549  *                                                                                             *
    550  * HISTORY:                                                                                    *
    551  *   07/18/1994 JLB : Created.                                                                 *
    552  *=============================================================================================*/
    553 int FootClass::Mission_Move(void)
    554 {
    555 	if (!Target_Legal(NavCom) && !IsDriving && MissionQueue == MISSION_NONE) {
    556 		Enter_Idle_Mode();
    557 	}
    558 	if (!Target_Legal(TarCom) && !House->IsHuman) {
    559 		Target_Something_Nearby(THREAT_RANGE);
    560 	}
    561 	return(TICKS_PER_SECOND+3);
    562 }
    563 
    564 
    565 /***********************************************************************************************
    566  * FootClass::Mission_Capture -- Handles the capture mission.                                  *
    567  *                                                                                             *
    568  *    Capture missions are nearly the same as normal movement missions. The only difference    *
    569  *    is that the final destination is handled in a special way so that it is not marked as    *
    570  *    impassable. This allows the object (usually infantry) the ability to walk onto the       *
    571  *    object and thus capture it.                                                              *
    572  *                                                                                             *
    573  * INPUT:   none                                                                               *
    574  *                                                                                             *
    575  * OUTPUT:  Returns with the number of game ticks to delay before calling this routine.        *
    576  *                                                                                             *
    577  * WARNINGS:   none                                                                            *
    578  *                                                                                             *
    579  * HISTORY:                                                                                    *
    580  *   03/19/1995 JLB : Created.                                                                 *
    581  *=============================================================================================*/
    582 int FootClass::Mission_Capture(void)
    583 {
    584 	if (!Target_Legal(NavCom) && !In_Radio_Contact()) {
    585 		Enter_Idle_Mode();
    586 		if (Map[Coord_Cell(Center_Coord())].Cell_Building()) {
    587 			Scatter(0, true);
    588 		}
    589 	}
    590 	return(TICKS_PER_SECOND-2);
    591 }
    592 
    593 
    594 /***********************************************************************************************
    595  * FootClass::Mission_Attack -- AI for heading towards and firing upon target.                 *
    596  *                                                                                             *
    597  *    This AI routine handles heading to within range of the target and then firing upon       *
    598  *    it until it is destroyed. If the target is destroyed, then the unit will change          *
    599  *    missions to match its "idle mode" of operation (usually guarding).                       *
    600  *                                                                                             *
    601  * INPUT:   none                                                                               *
    602  *                                                                                             *
    603  * OUTPUT:  Returns with the delay before calling this routine again.                          *
    604  *                                                                                             *
    605  * WARNINGS:   none                                                                            *
    606  *                                                                                             *
    607  * HISTORY:                                                                                    *
    608  *   07/18/1994 JLB : Created.                                                                 *
    609  *=============================================================================================*/
    610 int FootClass::Mission_Attack(void)
    611 {
    612 	if (Target_Legal(TarCom)) {
    613 		Approach_Target();
    614 	} else {
    615 		Enter_Idle_Mode();
    616 	}
    617 	return(TICKS_PER_SECOND+2);
    618 }
    619 
    620 
    621 /***********************************************************************************************
    622  * FootClass::Mission_Guard -- Handles the AI for guarding in place.                           *
    623  *                                                                                             *
    624  *    Units that are performing stationary guard duty use this AI process. They will sit       *
    625  *    still and target any enemies that get within range.                                      *
    626  *                                                                                             *
    627  * INPUT:   none                                                                               *
    628  *                                                                                             *
    629  * OUTPUT:  Returns with the delay before calling this routine again.                          *
    630  *                                                                                             *
    631  * WARNINGS:   none                                                                            *
    632  *                                                                                             *
    633  * HISTORY:                                                                                    *
    634  *   07/18/1994 JLB : Created.                                                                 *
    635  *=============================================================================================*/
    636 int FootClass::Mission_Guard(void)
    637 {
    638 	if (!Target_Something_Nearby(THREAT_RANGE)) {
    639 		Random_Animate();
    640 	}
    641 	return(TICKS_PER_SECOND+Random_Picky((int)0, (int)4, (char*)NULL, (int)0));
    642 }
    643 
    644 
    645 /***********************************************************************************************
    646  * FootClass::Mission_Hunt -- Handles the default hunt order.                                  *
    647  *                                                                                             *
    648  *    This routine is the default hunt order for game objects. It handles searching for a      *
    649  *    nearby object and heading toward it. The act of targetting will cause it to attack       *
    650  *    the target it selects.                                                                   *
    651  *                                                                                             *
    652  * INPUT:   none                                                                               *
    653  *                                                                                             *
    654  * OUTPUT:  Returns the game tick delay before calling this routine again.                     *
    655  *                                                                                             *
    656  * WARNINGS:   none                                                                            *
    657  *                                                                                             *
    658  * HISTORY:                                                                                    *
    659  *   10/17/1994 JLB : Created.                                                                 *
    660  *=============================================================================================*/
    661 int FootClass::Mission_Hunt(void)
    662 {
    663 	if (!Target_Something_Nearby(THREAT_NORMAL)) {
    664 		Random_Animate();
    665 	} else {
    666 		if (What_Am_I() == RTTI_INFANTRY && ((InfantryTypeClass const &)Class_Of()).Type == INFANTRY_E7) {
    667 			Assign_Destination(TarCom);
    668 			Assign_Mission(MISSION_CAPTURE);
    669 		} else {
    670 			Approach_Target();
    671 		}
    672 	}
    673 	return(TICKS_PER_SECOND+5);
    674 }
    675 
    676 
    677 /***********************************************************************************************
    678  * FootClass::Mission_Timed_Hunt -- This is the AI process for multiplayer computer units.     *
    679  *                                                                                             *
    680  * For multiplayer games, the computer AI can't just blitz the human players; the humans       *
    681  * need a little time to set up their base, or whatever.  This state just waits for            *
    682  * a certain period of time, then goes into hunt mode.                                         *
    683  *                                                                                             *
    684  * INPUT:   none                                                                               *
    685  *                                                                                             *
    686  * OUTPUT:  Returns with the delay before calling this routine again.                          *
    687  *                                                                                             *
    688  * WARNINGS:   none                                                                            *
    689  *                                                                                             *
    690  * HISTORY:                                                                                    *
    691  *   07/18/1994 JLB : Created.                                                                 *
    692  *=============================================================================================*/
    693 int FootClass::Mission_Timed_Hunt(void)
    694 {
    695 	int rndmax;
    696 	int changed = 0;					// has the unit changed into Hunt mode?
    697 
    698 	if (!House->IsHuman) {
    699 
    700 		/*
    701 		** Jump into HUNT mode if we're supposed to Blitz, and the EndCountDown
    702 		** has expired, or if our owning house has lost more than 1/4 of its units
    703 		** (it gets mad at you)
    704 		*/
    705 		if ( (MPlayerBlitz && House->BlitzTime==0) ||
    706 			House->CurUnits < ((House->MaxUnit * 4) / 5)) {
    707 			Assign_Mission(MISSION_HUNT);
    708 			changed = 1;
    709 		}
    710 
    711 		/*
    712 		** Jump into HUNT mode on a random die roll; the computer units will periodically
    713 		** "snap out" of their daze, and begin hunting.  Try to time it so that all
    714 		** units will be hunting within 10 minutes (600 calls to this routine).
    715 		*/
    716 		if (MPlayerBases) {
    717 			rndmax = 5000;
    718 		} else {
    719 			rndmax = 1000;
    720 		}
    721 
    722 		if (IRandom(0,rndmax) == 1) {
    723 			Assign_Mission(MISSION_HUNT);
    724 			changed = 1;
    725 		}
    726 
    727 		/*
    728 		** If this unit is still just sitting in Timed Hunt mode, call Guard Area
    729 		** so it doesn't just sit there stupidly.
    730 		*/
    731 		if (!changed) {
    732 			Mission_Guard_Area();
    733 		}
    734 	}
    735 
    736 	return(TICKS_PER_SECOND+Random_Pick(0, 4));		// call me back in 1 second.
    737 
    738 }
    739 
    740 
    741 /***********************************************************************************************
    742  * FootClass::Stop_Driver -- This routine clears the driving state of the object.              *
    743  *                                                                                             *
    744  *    This is the counterpart routine to the Start_Driver function. It clears the driving      *
    745  *    status flags and destination coordinate record.                                          *
    746  *                                                                                             *
    747  * INPUT:   none                                                                               *
    748  *                                                                                             *
    749  * OUTPUT:  bool; Was driving stopped?                                                         *
    750  *                                                                                             *
    751  * WARNINGS:   none                                                                            *
    752  *                                                                                             *
    753  * HISTORY:                                                                                    *
    754  *   10/17/1994 JLB : Created.                                                                 *
    755  *   12/12/1994 JLB : Greatly simplified.                                                      *
    756  *=============================================================================================*/
    757 bool FootClass::Stop_Driver(void)
    758 {
    759 	if (HeadToCoord) {
    760 		HeadToCoord = NULL;
    761 		Set_Speed(0);
    762 		IsDriving = false;
    763 		return(true);
    764 	}
    765 	return(false);
    766 }
    767 
    768 
    769 /***********************************************************************************************
    770  * FootClass::Start_Driver -- This starts the driver heading to the destination desired.       *
    771  *                                                                                             *
    772  *    Before a unit can move it must be started by this routine. This routine handles          *
    773  *    reserving the cell and setting the driving flag.                                         *
    774  *                                                                                             *
    775  * INPUT:   headto   -- The coordinate of the immediate drive destination. This is one cell    *
    776  *                      away from the unit's current location.                                 *
    777  *                                                                                             *
    778  * OUTPUT:  bool; Was driving initiated?                                                       *
    779  *                                                                                             *
    780  * WARNINGS:   none                                                                            *
    781  *                                                                                             *
    782  * HISTORY:                                                                                    *
    783  *   10/17/1994 JLB : Created.                                                                 *
    784  *   12/12/1994 JLB : Uses simple spot index finder.                                           *
    785  *=============================================================================================*/
    786 bool FootClass::Start_Driver(COORDINATE &headto)
    787 {
    788 	Stop_Driver();
    789 	if (headto) {
    790 		HeadToCoord = headto;
    791 		IsDriving = true;
    792 
    793 		/*
    794 		**	Check for crate goodie finder here.
    795 		*/
    796 		if (Map[Coord_Cell(headto)].Goodie_Check(this)) {
    797 			return(true);
    798 		}
    799 
    800 		HeadToCoord = NULL;
    801 		IsDriving = false;
    802 	}
    803 	return(false);
    804 }
    805 
    806 
    807 /***********************************************************************************************
    808  * FootClass::Sort_Y -- Determine the sort coordinate for foot class objects.                  *
    809  *                                                                                             *
    810  *    This routine will determine the sort coordinate for foot class object. This coordinate   *
    811  *    is usually the coordinate of the object. The exception is if the object is teathered.    *
    812  *    In this case (presumes offloading to the north), the sorting coordinate is adjusted      *
    813  *    so that the object will be drawn on top of the transport unit.                           *
    814  *                                                                                             *
    815  * INPUT:   none                                                                               *
    816  *                                                                                             *
    817  * OUTPUT:  Returns with the coordinate to use for sorting.                                    *
    818  *                                                                                             *
    819  * WARNINGS:   none                                                                            *
    820  *                                                                                             *
    821  * HISTORY:                                                                                    *
    822  *   10/17/1994 JLB : Created.                                                                 *
    823  *   11/04/1994 JLB : Sort value is different when unloading from aircraft.                    *
    824  *=============================================================================================*/
    825 COORDINATE FootClass::Sort_Y(void) const
    826 {
    827 	if (IsUnloading) {
    828 		return(Coord_Add(Coord, 0x01000000L));
    829 	}
    830 	if (In_Radio_Contact() && IsTethered && Contact_With_Whom()->What_Am_I() == RTTI_UNIT) {
    831 		return(Coord_Add(Coord, 0x01000000L));
    832 	}
    833 	return(Coord_Add(Coord, 0x00300000L));
    834 }
    835 
    836 
    837 /***********************************************************************************************
    838  * FootClass::Stun -- Prepares a ground travelling object for removal.                         *
    839  *                                                                                             *
    840  *    This routine clears the units' navigation computer in preparation for removal from the   *
    841  *    game. This is probably called as a result of unit destruction in combat. Clearing the    *
    842  *    navigation computer ensures that the normal AI process won't start it moving again while *
    843  *    the object is undergoing any death animations.                                           *
    844  *                                                                                             *
    845  * INPUT:   none                                                                               *
    846  *                                                                                             *
    847  * OUTPUT:  none                                                                               *
    848  *                                                                                             *
    849  * WARNINGS:   none                                                                            *
    850  *                                                                                             *
    851  * HISTORY:                                                                                    *
    852  *   12/23/1994 JLB : Created.                                                                 *
    853  *=============================================================================================*/
    854 void FootClass::Stun(void)
    855 {
    856 	Assign_Destination(TARGET_NONE);
    857 	Path[0] = FACING_NONE;
    858 	Stop_Driver();
    859 	TechnoClass::Stun();
    860 }
    861 
    862 
    863 /***********************************************************************************************
    864  * FootClass::Approach_Target -- Sets the navigation computer to approach target object.       *
    865  *                                                                                             *
    866  *    This routine will set the navigation computer to approach the target indicated by the    *
    867  *    targeting computer. It is through this function that the unit nears the target so        *
    868  *    that weapon firing may occur.                                                            *
    869  *                                                                                             *
    870  * INPUT:   none                                                                               *
    871  *                                                                                             *
    872  * OUTPUT:  none                                                                               *
    873  *                                                                                             *
    874  * WARNINGS:   none                                                                            *
    875  *                                                                                             *
    876  * HISTORY:                                                                                    *
    877  *   05/31/1994 JLB : Created.                                                                 *
    878  *   12/13/1994 JLB : Made part of TechnoClass.                                                *
    879  *   12/22/1994 JLB : Enhanced search algorithm.                                               *
    880  *   05/20/1995 JLB : Always approaches if the object is off the map.                          *
    881  *=============================================================================================*/
    882 void FootClass::Approach_Target(void)
    883 {
    884 	/*
    885 	**	Determine that if there is an existing target it is still legal
    886 	**	and within range.
    887 	*/
    888 	if (Target_Legal(TarCom)) {
    889 
    890 		/*
    891 		**	If the target is too far away then head toward it.
    892 		*/
    893 		int maxrange = MAX(Weapon_Range(0), Weapon_Range(1));
    894 
    895 		if (!Target_Legal(NavCom) && (!In_Range(TarCom) || !IsLocked)) {
    896 //		if (!Target_Legal(NavCom) && (Distance(TarCom) > maxrange || !IsLocked)) {
    897 
    898 			/*
    899 			** If the object that we are attacking is a building adjust the units
    900 			** max range so that people can stand far away from the buildings and
    901 			** hit them.
    902 			*/
    903 			BuildingClass * obj = As_Building(TarCom);
    904 			if (obj) {
    905 				maxrange += ((obj->Class->Width() + obj->Class->Height()) * (0x100 / 4));
    906 			}
    907 
    908 			/*
    909 			** Adjust the max range of an infantry unit for where he is standing
    910 			** in the room.
    911 			*/
    912 			maxrange -= 0x00B7;
    913 #ifdef OBSOLETE
    914 			if (What_Am_I() == RTTI_INFANTRY) {
    915 				maxrange -= 0x0111;
    916 			} else {
    917 				maxrange -= 0x00B7;
    918 			}
    919 #endif
    920 			maxrange = MAX(maxrange, 0);
    921 
    922 			COORDINATE tcoord = ::As_Coord(TarCom);
    923 			COORDINATE trycoord = 0;
    924 			CELL tcell = Coord_Cell(tcoord);
    925 			CELL trycell = tcell;
    926 			DirType dir = Direction256(tcoord, Center_Coord());
    927 			bool found = false;
    928 
    929 			/*
    930 			**	Sweep through the cells between the target and the unit, looking for
    931 			**	a cell that the unit can enter but which is also within weapon range
    932 			**	of the target. If after a reasonable search, no appropriate cell could
    933 			**	be found, then the target will be assigned as the movement destination
    934 			**	and let "the chips fall where they may."
    935 			*/
    936 			for (int range = maxrange; range > 0x0080; range -= 0x0100) {
    937 				static int _angles[] = {0, 8, -8, 16, -16, 24, -24, 32, -32, 48, -48, 64, -64};
    938 
    939 				for (int index = 0; index < (sizeof(_angles)/sizeof(_angles[0])); index++) {
    940 					trycoord = Coord_Move(tcoord, (DirType)(dir + _angles[index]), range);
    941 
    942 					if (::Distance(trycoord, tcoord) < range) {
    943 						trycell = Coord_Cell(trycoord);
    944 						if (Can_Enter_Cell(trycell) <= MOVE_CLOAK && Map.In_Radar(trycell)) {
    945 							found = true;
    946 							break;
    947 						}
    948 					}
    949 				}
    950 				if (found) break;
    951 			}
    952 
    953 			/*
    954 			**	If a suitable intermediate location was found, then head toward it.
    955 			**	Otherwise, head toward the enemy unit directly.
    956 			**	Infantry always head towards the target since they can enter a cell
    957 			**	in range, but still not be able to hit the target if the spot is out of range.
    958 			*/
    959 			if (found && What_Am_I() != RTTI_INFANTRY) {
    960 				Assign_Destination(::As_Target(trycell));
    961 			} else {
    962 				Assign_Destination(TarCom);
    963 			}
    964 		}
    965 	}
    966 }
    967 
    968 
    969 /***********************************************************************************************
    970  * FootClass::Mission_Guard_Area -- Causes unit to guard an area about twice weapon range.     *
    971  *                                                                                             *
    972  *    This mission routine causes the unit to scan for targets out to twice its weapon range   *
    973  *    from the home point. If a target was found, then it will be attacked. The unit will      *
    974  *    chase the target until it gets up to to its weapon range from the home position.         *
    975  *    In that case, it will return to home position and start scanning for another target.     *
    976  *                                                                                             *
    977  * INPUT:   none                                                                               *
    978  *                                                                                             *
    979  * OUTPUT:  Returns with time delay before calling this routine again.                         *
    980  *                                                                                             *
    981  * WARNINGS:   none                                                                            *
    982  *                                                                                             *
    983  * HISTORY:                                                                                    *
    984  *   12/23/1994 JLB : Created.                                                                 *
    985  *   07/27/1995 JLB : Greatly simplified.                                                      *
    986  *=============================================================================================*/
    987 int FootClass::Mission_Guard_Area(void)
    988 {
    989 	if (What_Am_I() == RTTI_UNIT && ((UnitClass *)this)->Class->IsToHarvest) {
    990 		Assign_Mission(MISSION_HARVEST);
    991 		return(1+Random_Pick(1, 10));
    992 	}
    993 
    994 	/*
    995 	**	Ensure that the archive target is valid.
    996 	*/
    997 	if (!Target_Legal(ArchiveTarget)) {
    998 		ArchiveTarget = ::As_Target(Coord_Cell(Coord));
    999 	}
   1000 
   1001 	/*
   1002 	**	Ensure units aren't trying to guard cells off the map.
   1003 	*/
   1004 	if (Target_Legal(NavCom) && Is_Target_Cell(NavCom)) {
   1005 		CELL cell = As_Cell(NavCom);
   1006 		int x = Cell_X(cell);
   1007 		int y = Cell_Y(cell);
   1008 		if (x < Map.MapCellX || y < Map.MapCellY || x >= (Map.MapCellX + Map.MapCellWidth) || y >= (Map.MapCellY + Map.MapCellHeight)) {
   1009 			Assign_Target(TARGET_NONE);
   1010 			Assign_Destination(TARGET_NONE);
   1011 			ArchiveTarget = ::As_Target(Coord_Cell(Coord));
   1012 		}
   1013 	}
   1014 
   1015 	/*
   1016 	**	Make sure that the unit has not strayed too far from the home position.
   1017 	**	If it has, then race back to it.
   1018 	*/
   1019 	int maxrange = MAX(Weapon_Range(0), Weapon_Range(1))+0x0100;
   1020 	if (!Target_Legal(NavCom) && (Distance(ArchiveTarget) > maxrange || (!Target_Legal(TarCom) && Distance(ArchiveTarget) > 0x0200))) {
   1021 		Assign_Target(TARGET_NONE);
   1022 		Assign_Destination(ArchiveTarget);
   1023 	}
   1024 
   1025 	if (!Target_Legal(TarCom)) {
   1026 		COORDINATE old = Coord;
   1027 		Coord = As_Coord(ArchiveTarget);
   1028 		Target_Something_Nearby(THREAT_AREA);
   1029 		Coord = old;
   1030 		if (Target_Legal(TarCom)) return(1);
   1031 		Random_Animate();
   1032 	} else {
   1033 		Approach_Target();
   1034 	}
   1035 	return(TICKS_PER_SECOND+Random_Picky((int)0, (int)4, (char*)NULL, (int)0));
   1036 }
   1037 
   1038 
   1039 /***********************************************************************************************
   1040  * FootClass::Unlimbo -- Unlimbos object and performs special fixups.                          *
   1041  *                                                                                             *
   1042  *    This routine will make sure that the home position for the foot class object gets        *
   1043  *    reset. This is necessary since the home position may change depending on the unit's      *
   1044  *    transition between limbo and non-limbo condition.                                        *
   1045  *                                                                                             *
   1046  * INPUT:   coord    -- The coordinate to unlimbo the unit at.                                 *
   1047  *                                                                                             *
   1048  *          dir      -- The initial direction to give the unit.                                *
   1049  *                                                                                             *
   1050  * OUTPUT:  bool; Was the unit unlimboed successfully?                                         *
   1051  *                                                                                             *
   1052  * WARNINGS:   none                                                                            *
   1053  *                                                                                             *
   1054  * HISTORY:                                                                                    *
   1055  *   12/23/1994 JLB : Created.                                                                 *
   1056  *=============================================================================================*/
   1057 bool FootClass::Unlimbo(COORDINATE coord, DirType dir)
   1058 {
   1059 	/*
   1060 	**	Try to unlimbo the unit.
   1061 	*/
   1062 	if (TechnoClass::Unlimbo(coord, dir)) {
   1063 
   1064 		/*
   1065 		**	Mobile units are always revealed to the house that owns them.
   1066 		*/
   1067 		Revealed(House);
   1068 
   1069 		/*
   1070 		**	Start in a still (non-moving) state.
   1071 		*/
   1072 		Path[0] = FACING_NONE;
   1073 		return(true);
   1074 	}
   1075 	return(false);
   1076 }
   1077 
   1078 
   1079 /***********************************************************************************************
   1080  * FootClass::Assign_Mission -- Assign mission to foot class object.                           *
   1081  *                                                                                             *
   1082  *    When a new mission is assigned, any precalculated path should be truncated. This is      *
   1083  *    in anticipation that the new mission will result in a change of path.                    *
   1084  *                                                                                             *
   1085  * INPUT:   order -- The new mission to assign to the unit.                                    *
   1086  *                                                                                             *
   1087  * OUTPUT:  none                                                                               *
   1088  *                                                                                             *
   1089  * WARNINGS:   none                                                                            *
   1090  *                                                                                             *
   1091  * HISTORY:                                                                                    *
   1092  *   12/29/1994 JLB : Created.                                                                 *
   1093  *=============================================================================================*/
   1094 void FootClass::Assign_Mission(MissionType order)
   1095 {
   1096 	if (What_Am_I() != RTTI_UNIT || *(UnitClass*)this != UNIT_GUNBOAT) {
   1097 		Path[0] = FACING_NONE;
   1098 	}
   1099 	TechnoClass::Assign_Mission(order);
   1100 }
   1101 
   1102 
   1103 /***********************************************************************************************
   1104  * FootClass::Limbo -- Intercepts limbo event and handles FootClass processing.                *
   1105  *                                                                                             *
   1106  *    When an object of FootClass type is limboed, then it must be removed from any team       *
   1107  *    it may be a member of.                                                                   *
   1108  *                                                                                             *
   1109  * INPUT:   none                                                                               *
   1110  *                                                                                             *
   1111  * OUTPUT:  bool; Was the object successfully limboed?                                         *
   1112  *                                                                                             *
   1113  * WARNINGS:   none                                                                            *
   1114  *                                                                                             *
   1115  * HISTORY:                                                                                    *
   1116  *   12/29/1994 JLB : Created.                                                                 *
   1117  *=============================================================================================*/
   1118 bool FootClass::Limbo(void)
   1119 {
   1120 	if (!IsInLimbo) {
   1121 		if (Team) {
   1122 			Team->Remove(this);
   1123 		}
   1124 	}
   1125 	return(TechnoClass::Limbo());
   1126 }
   1127 
   1128 
   1129 /***********************************************************************************************
   1130  * FootClass::Take_Damage -- Handles taking damage to this object.                             *
   1131  *                                                                                             *
   1132  *    This routine intercepts the damage assigned to this object and if this object is         *
   1133  *    a member of a team, it informs the team that the damage has occurred. The team may       *
   1134  *    change it's priority or action based on this event.                                      *
   1135  *                                                                                             *
   1136  * INPUT:   damage      -- The damage points inflicted on the unit.                            *
   1137  *                                                                                             *
   1138  *          distance    -- The distance from the point of damage to the unit itself.           *
   1139  *                                                                                             *
   1140  *          warhead     -- The type of damage that is inflicted.                               *
   1141  *                                                                                             *
   1142  *          source      -- The purpitrator of the damage. By knowing who caused the damage,    *
   1143  *                         the team know's who to "get even with".                             *
   1144  *                                                                                             *
   1145  * OUTPUT:  Returns with the result type of the damage.                                        *
   1146  *                                                                                             *
   1147  * WARNINGS:   none                                                                            *
   1148  *                                                                                             *
   1149  * HISTORY:                                                                                    *
   1150  *   12/30/1994 JLB : Created.                                                                 *
   1151  *=============================================================================================*/
   1152 ResultType FootClass::Take_Damage(int & damage, int distance, WarheadType warhead, TechnoClass * source)
   1153 {
   1154 	ResultType result = TechnoClass::Take_Damage(damage, distance, warhead, source);
   1155 
   1156 	if (result != RESULT_NONE && Team) {
   1157 
   1158 		if (GameToPlay != GAME_NORMAL || (source && !House->Is_Ally(source))) {
   1159 			Team->Took_Damage(this, result, source);
   1160 		}
   1161 
   1162 	} else {
   1163 
   1164 		if (result != RESULT_DESTROYED) {
   1165 			/*
   1166 			**	Determine if the target that is current being attacked has a weapon that can
   1167 			**	do harm to a ground based unit. This information is needed so that an appropriate
   1168 			**	response will occur when damage is taken.
   1169 			*/
   1170 			WeaponType weap = WEAPON_NONE;
   1171 			if (As_Techno(TarCom)) {
   1172 				weap = As_Techno(TarCom)->Techno_Type_Class()->Primary;
   1173 			}
   1174 			bool tweap = (weap != WEAPON_NONE && weap != WEAPON_NIKE);
   1175 
   1176 			/*
   1177 			**	This ensures that if a unit is in sticky mode, then it will snap out of
   1178 			**	it when it takes damage.
   1179 			*/
   1180 			if (source && Mission == MISSION_STICKY) {
   1181 				Enter_Idle_Mode();
   1182 			}
   1183 
   1184 			/*
   1185 			**	If this object is not part of a team and it can retaliate for the damage, then have
   1186 			**	it try to do so. This prevents it from just sitting there and taking damage.
   1187 			*/
   1188 			if (
   1189 				source &&
   1190 				!House->Is_Ally(source) &&
   1191 				Techno_Type_Class()->Primary != WEAPON_NONE &&
   1192 				(source->What_Am_I() != RTTI_AIRCRAFT || BulletTypeClass::As_Reference(Weapons[Techno_Type_Class()->Primary].Fires).IsAntiAircraft) &&
   1193 				(!Target_Legal(TarCom) || ((!House->IsHuman || Special.IsSmartDefense) && (!tweap || !In_Range(TarCom)))) &&
   1194 //				!Target_Legal(NavCom) &&
   1195 					(Mission == MISSION_AMBUSH ||
   1196 					Mission == MISSION_GUARD ||
   1197 					Mission == MISSION_RESCUE ||
   1198 					Mission == MISSION_GUARD_AREA ||
   1199 					Mission == MISSION_ATTACK ||
   1200 					Mission == MISSION_TIMED_HUNT)) {
   1201 
   1202 				/*
   1203 				**	Assign the source of the damage as the new target. This occurs for the computer
   1204 				**	controled units. For the player, this only occurs if the source of the damage
   1205 				**	is within range.
   1206 				*/
   1207 				if (!House->IsHuman) {
   1208 
   1209 					/*
   1210 					** If this unit is in TIMED_HUNT (multiplayer computer-controlled)
   1211 					** mode, "snap out of it" into HUNT mode; otherwise, assign
   1212 					** HUNT as the next mission through the normal mission queue.
   1213 					*/
   1214 					if (Mission == MISSION_TIMED_HUNT) {
   1215 						Set_Mission(MISSION_HUNT);
   1216 					} else {
   1217 						Assign_Mission(MISSION_HUNT);
   1218 					}
   1219 					Assign_Target(source->As_Target());
   1220 				} else {
   1221 					if (In_Range(source)) {
   1222 						Assign_Target(source->As_Target());
   1223 					} else {
   1224 
   1225 						/*
   1226 						**	Simple retaliation cannot occur because the source of the damage
   1227 						**	is too far away. If scatter logic is enabled, then scatter now.
   1228 						*/
   1229 						if (Special.IsScatter) {
   1230 							Scatter(0, true);
   1231 						}
   1232 					}
   1233 				}
   1234 			} else {
   1235 
   1236 				/*
   1237 				**	If this object isn't doing anything important, then scatter.
   1238 				*/
   1239 				if (!IsDriving && !Target_Legal(TarCom) && !Target_Legal(NavCom) && Special.IsScatter && What_Am_I() != RTTI_AIRCRAFT) {
   1240 					Scatter(0, true);
   1241 				}
   1242 			}
   1243 		}
   1244 	}
   1245 	return(result);
   1246 }
   1247 
   1248 
   1249 /***********************************************************************************************
   1250  * FootClass::Active_Click_With -- Intiates attack or move according to target clicked on.     *
   1251  *                                                                                             *
   1252  *    At this level, the object is known to have the ability to attack or move to the          *
   1253  *    target specified (in theory). Perform the attack or move as indicated.                   *
   1254  *                                                                                             *
   1255  * INPUT:   target   -- The target clicked upon that will precipitate action.                  *
   1256  *                                                                                             *
   1257  * OUTPUT:  Returns with the type of action performed.                                         *
   1258  *                                                                                             *
   1259  * WARNINGS:   none                                                                            *
   1260  *                                                                                             *
   1261  * HISTORY:                                                                                    *
   1262  *   01/06/1995 JLB : Created.                                                                 *
   1263  *=============================================================================================*/
   1264 void FootClass::Active_Click_With(ActionType action, ObjectClass * object)
   1265 {
   1266 	switch (action) {
   1267 		case ACTION_GUARD_AREA:
   1268 			if (Can_Player_Fire() && Can_Player_Move()) {
   1269 				Player_Assign_Mission(MISSION_GUARD_AREA, object->As_Target());
   1270 			}
   1271 			break;
   1272 
   1273 		case ACTION_SELF:
   1274 			Player_Assign_Mission(MISSION_UNLOAD);
   1275 			break;
   1276 
   1277 		case ACTION_ATTACK:
   1278 			if (Can_Player_Fire()) {
   1279 				Player_Assign_Mission(MISSION_ATTACK, object->As_Target());
   1280 			}
   1281 			break;
   1282 
   1283 		case ACTION_ENTER:
   1284 			if (Can_Player_Move() && object && object->Is_Techno() /*&& !((RadioClass *)object)->In_Radio_Contact()*/) {
   1285 				Player_Assign_Mission(MISSION_ENTER, TARGET_NONE, object->As_Target());
   1286 			}
   1287 			break;
   1288 
   1289 		case ACTION_CAPTURE:
   1290 			if (Can_Player_Move()) {
   1291 				Player_Assign_Mission(MISSION_CAPTURE, TARGET_NONE, object->As_Target());
   1292 			}
   1293 			break;
   1294 
   1295 		case ACTION_SABOTAGE:
   1296 			if (Can_Player_Move()) {
   1297 				Player_Assign_Mission(MISSION_SABOTAGE, TARGET_NONE, object->As_Target());
   1298 			}
   1299 			break;
   1300 
   1301 		case ACTION_NOMOVE:
   1302 		case ACTION_MOVE:
   1303 			if (Can_Player_Move()) {
   1304 				Player_Assign_Mission(MISSION_MOVE, TARGET_NONE, object->As_Target());
   1305 			}
   1306 			break;
   1307 
   1308 		case ACTION_NO_DEPLOY:
   1309 			Speak(VOX_DEPLOY);
   1310 			break;
   1311 
   1312 		default:
   1313 			break;
   1314 	}
   1315 }
   1316 
   1317 
   1318 /***********************************************************************************************
   1319  * FootClass::Active_Click_With -- Performs action as a result of left mouse click.            *
   1320  *                                                                                             *
   1321  *    This routine performs the action requested when the left mouse button was clicked over   *
   1322  *    a cell. Typically, this is just a move command.                                          *
   1323  *                                                                                             *
   1324  * INPUT:   action   -- The predetermined action that should occur.                            *
   1325  *                                                                                             *
   1326  *          cell     -- The cell number that the action should occur at.                       *
   1327  *                                                                                             *
   1328  * OUTPUT:  none                                                                               *
   1329  *                                                                                             *
   1330  * WARNINGS:   none                                                                            *
   1331  *                                                                                             *
   1332  * HISTORY:                                                                                    *
   1333  *   01/19/1995 JLB : Created.                                                                 *
   1334  *=============================================================================================*/
   1335 void FootClass::Active_Click_With(ActionType action, CELL cell)
   1336 {
   1337 	switch (action) {
   1338 		case ACTION_HARVEST:
   1339 			Player_Assign_Mission(MISSION_HARVEST, TARGET_NONE, ::As_Target(cell));
   1340 			break;
   1341 
   1342 		case ACTION_MOVE:
   1343 			if (AllowVoice) {
   1344 				COORDINATE coord = Map.Pixel_To_Coord(Get_Mouse_X(), Get_Mouse_Y());
   1345 				OutList.Add(EventClass(ANIM_MOVE_FLASH, PlayerPtr->Class->House, coord, 1 << PlayerPtr->Class->House));
   1346 			}
   1347 			// Fall into next case.
   1348 
   1349 		case ACTION_NOMOVE:
   1350 			//using function for IsVisible so we have different results for different players - JAS 2019/09/30
   1351 			if (What_Am_I() != RTTI_AIRCRAFT || Map[cell].Is_Visible(PlayerPtr)) {
   1352 				Player_Assign_Mission(MISSION_MOVE, TARGET_NONE, ::As_Target(cell));
   1353 			}
   1354 			break;
   1355 
   1356 		case ACTION_ATTACK:
   1357 			Player_Assign_Mission(MISSION_ATTACK, ::As_Target(cell));
   1358 			break;
   1359 
   1360 		// MBL 05.15.2020 - Adding support for CTRL+ALT clicking the ground to have units move to an area and guard it
   1361 		case ACTION_GUARD_AREA:
   1362 			if (Can_Player_Fire() && Can_Player_Move()) {
   1363 				Player_Assign_Mission(MISSION_GUARD_AREA, ::As_Target(cell));
   1364 			}
   1365 			break;
   1366 		// END MBL 05.15.2020 
   1367 	}
   1368 }
   1369 
   1370 
   1371 /***********************************************************************************************
   1372  * FootClass::Per_Cell_Process -- Perform action based on once-per-cell condition.             *
   1373  *                                                                                             *
   1374  *    This routine is called as this object moves from cell to cell. When the center of the    *
   1375  *    cell is reached, check to see if any trigger should be sprung. For moving units, reduce  *
   1376  *    the path to the distance to the target. This forces path recalculation in an effort to   *
   1377  *    avoid units passing each other.                                                          *
   1378  *                                                                                             *
   1379  * INPUT:   center   -- Is this the center of the cell?                                        *
   1380  *                                                                                             *
   1381  * OUTPUT:  none                                                                               *
   1382  *                                                                                             *
   1383  * WARNINGS:   none                                                                            *
   1384  *                                                                                             *
   1385  * HISTORY:                                                                                    *
   1386  *   05/08/1995 JLB : Created.                                                                 *
   1387  *   07/08/1995 JLB : Handles generic enter trigger event.                                     *
   1388  *   07/16/1995 JLB : If next to a scanner and cloaked, then shimmer.                          *
   1389  *=============================================================================================*/
   1390 void FootClass::Per_Cell_Process(bool center)
   1391 {
   1392 //	if (center) {
   1393 
   1394 		/*
   1395 		**	Clear any unloading flag if necessary.
   1396 		*/
   1397 		IsUnloading = false;
   1398 
   1399 		/*
   1400 		**	If adjacent to an enemy building that has the ability to reveal a stealth tank,
   1401 		**	then shimmer the cloaked object.
   1402 		*/
   1403 		if (Cloak == CLOAKED) {
   1404 			for (FacingType face = FACING_N; face < FACING_COUNT; face++) {
   1405 				CELL cell = Adjacent_Cell(Coord_Cell(Coord), face);
   1406 
   1407 				if (Map.In_Radar(cell)) {
   1408 					TechnoClass const * techno = Map[cell].Cell_Techno();
   1409 
   1410 					if (techno && !House->Is_Ally(techno) && techno->Techno_Type_Class()->IsScanner) {
   1411 						Do_Shimmer();
   1412 						break;
   1413 					}
   1414 				}
   1415 			}
   1416 		}
   1417 
   1418 		/*
   1419 		**	Shorten the path if the target is now within weapon range of this
   1420 		**	unit and this unit is on an attack type mission. But only if the target
   1421 		**	is slow enough for leading to make sense.
   1422 		*/
   1423 		if (What_Am_I() != RTTI_UNIT || *((UnitClass *)this) != UNIT_GUNBOAT) {
   1424 			bool inrange = In_Range(TarCom);
   1425 			TechnoClass const * techno = As_Techno(TarCom);
   1426 			if (techno && techno->What_Am_I() != RTTI_BUILDING) {
   1427 				FootClass const * foot = (FootClass const *)techno;
   1428 				MPHType speed = ((TechnoTypeClass const &)techno->Class_Of()).MaxSpeed;
   1429 				COORDINATE rangecoord = (speed > MPH_SLOW) ? foot->Likely_Coord() : foot->Target_Coord();
   1430 				inrange = In_Range(rangecoord);
   1431 			}
   1432 
   1433 			if (Target_Legal(TarCom) && (Mission == MISSION_RESCUE || Mission == MISSION_GUARD_AREA || Mission == MISSION_ATTACK || Mission == MISSION_HUNT) && inrange) {
   1434 				Assign_Destination(TARGET_NONE);
   1435 				Path[0] = FACING_NONE;
   1436 			}
   1437 		}
   1438 
   1439 		/*
   1440 		**	Trigger event associated with the player entering the cell.
   1441 		*/
   1442 		TriggerClass * trigger = Map[Coord_Cell(Coord)].Get_Trigger();
   1443 		if (Cloak != CLOAKED && trigger && trigger->House == Owner()) {
   1444 			trigger->Spring(EVENT_PLAYER_ENTERED, Coord_Cell(Coord));
   1445 		}
   1446 //	}
   1447 
   1448 	Map[Coord_Cell(Coord)].Goodie_Check(this, true);
   1449 
   1450 	TechnoClass::Per_Cell_Process(center);
   1451 }
   1452 
   1453 
   1454 /***************************************************************************
   1455  * FootClass::Override_Mission -- temporarily overides a units mission     *
   1456  *                                                                         *
   1457  *                                                                         *
   1458  *                                                                         *
   1459  * INPUT:		MissionType mission - the mission we want to overide        *
   1460  *					TARGET	   tarcom  - the new target we want to overide		*
   1461  *					TARGET		navcom  - the new navigation point to overide	*
   1462  *                                                                         *
   1463  * OUTPUT:		none                                                        *
   1464  *                                                                         *
   1465  * WARNINGS:   If a mission is already overidden, the current mission is   *
   1466  *					just re-assigned.															*
   1467  *                                                                         *
   1468  * HISTORY:                                                                *
   1469  *   04/28/1995 PWG : Created.                                             *
   1470  *=========================================================================*/
   1471 void FootClass::Override_Mission(MissionType mission, TARGET tarcom, TARGET navcom)
   1472 {
   1473  	SuspendedNavCom = NavCom;
   1474 	TechnoClass::Override_Mission(mission, tarcom, navcom);
   1475 
   1476 	Assign_Destination(navcom);
   1477 }
   1478 
   1479 
   1480 /***************************************************************************
   1481  * FootClass::Restore_Mission -- Restores an overidden mission             *
   1482  *                                                                         *
   1483  * INPUT:		none                                                        *
   1484  *                                                                         *
   1485  * OUTPUT:     none                                                        *
   1486  *                                                                         *
   1487  * WARNINGS:   none                                                        *
   1488  *                                                                         *
   1489  * HISTORY:                                                                *
   1490  *   04/28/1995 PWG : Created.                                             *
   1491  *=========================================================================*/
   1492 bool FootClass::Restore_Mission(void)
   1493 {
   1494 	if (TechnoClass::Restore_Mission()) {
   1495 		Assign_Destination(SuspendedNavCom);
   1496 		return(true);
   1497 	}
   1498 	return(false);
   1499 }
   1500 
   1501 
   1502 /***********************************************************************************************
   1503  * FootClass::Receive_Message -- Movement related radio messages are handled here.             *
   1504  *                                                                                             *
   1505  *    This routine handles radio message that are related to movement. These are used for      *
   1506  *    complex coordinated maneuvers.                                                           *
   1507  *                                                                                             *
   1508  * INPUT:   from     -- Pointer to the originator of this radio message.                       *
   1509  *                                                                                             *
   1510  *          message  -- The radio message that is being received.                              *
   1511  *                                                                                             *
   1512  *          param    -- The optional parameter (could be a movement destination).              *
   1513  *                                                                                             *
   1514  * OUTPUT:  Returns with the radio response appropriate to the message received. Usually the   *
   1515  *          response is RADIO_ROGER.                                                           *
   1516  *                                                                                             *
   1517  * WARNINGS:   none                                                                            *
   1518  *                                                                                             *
   1519  * HISTORY:                                                                                    *
   1520  *   05/14/1995 JLB : Created.                                                                 *
   1521  *=============================================================================================*/
   1522 RadioMessageType FootClass::Receive_Message(RadioClass * from, RadioMessageType message, long & param)
   1523 {
   1524 	switch (message) {
   1525 
   1526 		/*
   1527 		**	Answers if this object is located on top of a service depot.
   1528 		*/
   1529 		case RADIO_ON_DEPOT:
   1530 			if (Map[Coord_Cell(Center_Coord())].Cell_Building() != NULL) {
   1531 				BuildingClass const * building = Map[Coord_Cell(Center_Coord())].Cell_Building();
   1532 				if (*building == STRUCT_REPAIR) {
   1533 					return(RADIO_ROGER);
   1534 				}
   1535 			}
   1536 			return(RADIO_NEGATIVE);
   1537 
   1538 		/*
   1539 		**	Intercept the repair request and if this object is moving, then no repair
   1540 		**	is possible.
   1541 		*/
   1542 		case RADIO_REPAIR:
   1543 			if (Target_Legal(NavCom)) return(RADIO_NEGATIVE);
   1544 			break;
   1545 
   1546 		/*
   1547 		**	Something bad has happened to the object in contact with. Abort any coordinated
   1548 		**	activity with this object. Basically, ... run away! Run away!
   1549 		*/
   1550 		case RADIO_RUN_AWAY:
   1551 			if (In_Radio_Contact()) {
   1552 				if (NavCom == Contact_With_Whom()->As_Target()) {
   1553 					Assign_Destination(TARGET_NONE);
   1554 				}
   1555 			}
   1556 			if (Mission == MISSION_SLEEP) {
   1557 				Assign_Mission(MISSION_GUARD);
   1558 				Commence();
   1559 			}
   1560 			if (Mission == MISSION_ENTER) {
   1561 				Assign_Mission(MISSION_GUARD);
   1562 			}
   1563 			if (!IsRotating && !Target_Legal(NavCom)) {
   1564 				Scatter(0, true, true);
   1565 			}
   1566 			break;
   1567 
   1568 		/*
   1569 		**	Checks to see if this unit needs to move somewhere. If it is already in motion,
   1570 		**	then it doesn't need furthur movement instructions.
   1571 		*/
   1572 		case RADIO_NEED_TO_MOVE:
   1573 			param = (long)NavCom;
   1574 			if (!Target_Legal(NavCom)) {
   1575 				return(RADIO_ROGER);
   1576 			}
   1577 			return(RADIO_NEGATIVE);
   1578 
   1579 		/*
   1580 		**	Radio request to move to location specified. Typically this is used
   1581 		**	for complex loading and unloading missions.
   1582 		*/
   1583 		case RADIO_MOVE_HERE:
   1584 			if (NavCom != (TARGET)param) {
   1585 				if (::As_Target(Coord_Cell(Coord)) == (TARGET)param) {
   1586 					return(RADIO_YEA_NOW_WHAT);
   1587 				} else {
   1588 					if (Mission == MISSION_GUARD && MissionQueue == MISSION_NONE) {
   1589 						Assign_Mission(MISSION_MOVE);
   1590 					}
   1591 					Assign_Destination((TARGET)param);
   1592 				}
   1593 			}
   1594 			return(RADIO_ROGER);
   1595 
   1596 		/*
   1597 		** Requests if this unit is trying to cooperatively load up. Typically, this occurs
   1598 		**	for passengers and when vehicles need to be repaired.
   1599 		*/
   1600 		case RADIO_TRYING_TO_LOAD:
   1601 			if (Mission == MISSION_ENTER || MissionQueue == MISSION_ENTER) {
   1602 				TechnoClass::Receive_Message(from, message, param);
   1603 				return(RADIO_ROGER);
   1604 			}
   1605 			break;
   1606 	}
   1607 	return(TechnoClass::Receive_Message(from, message, param));
   1608 }
   1609 
   1610 
   1611 /***********************************************************************************************
   1612  * FootClass::Mission_Enter -- Enter (cooperatively) mission handler.                          *
   1613  *                                                                                             *
   1614  *    This mission handler will cooperatively coordinate the object to maneuver into the       *
   1615  *    object it is in radio contact with. This is used by infantry when they wish to load      *
   1616  *    into an APC as well as by vehicles when they wish to enter a repair facility.            *
   1617  *                                                                                             *
   1618  * INPUT:   none                                                                               *
   1619  *                                                                                             *
   1620  * OUTPUT:  Returns the number of game ticks before this routine should be called again.       *
   1621  *                                                                                             *
   1622  * WARNINGS:   none                                                                            *
   1623  *                                                                                             *
   1624  * HISTORY:                                                                                    *
   1625  *   05/15/1995 JLB : Created.                                                                 *
   1626  *=============================================================================================*/
   1627 int FootClass::Mission_Enter(void)
   1628 {
   1629 	/*
   1630 	**	Find out who to coordinate with. If in radio contact, then this the transporter is
   1631 	**	defined. If not in radio contact, then try the archive target value to see if that
   1632 	**	is suitable.
   1633 	*/
   1634 	TechnoClass * contact = Contact_With_Whom();
   1635 	if (contact == NULL) {
   1636 		contact = As_Techno(ArchiveTarget);
   1637 	}
   1638 	if (contact == NULL) {
   1639 		contact = As_Techno(NavCom);
   1640 	}
   1641 
   1642 	/*
   1643 	**	If in contact, then let the transporter handle the movement coordination.
   1644 	*/
   1645 	if (contact != NULL) {
   1646 
   1647 		/*
   1648 		**	If the transport says to "bug off", then abort the enter mission. The transport may
   1649 		**	likely say all is 'ok' with the "RADIO ROGER", then try again later.
   1650 		*/
   1651 		if (Transmit_Message(RADIO_DOCKING, contact) != RADIO_ROGER && !IsTethered) {
   1652 			Transmit_Message(RADIO_OVER_OUT);
   1653 			Enter_Idle_Mode();
   1654 		}
   1655 
   1656 	} else {
   1657 
   1658 		/*
   1659 		**	Since there is no potential object to enter, then abort this
   1660 		**	mission with some default standby mission.
   1661 		*/
   1662 		if (MissionQueue == MISSION_NONE) {
   1663 			/*
   1664 			**	If this is a harvester, then return to harvesting.
   1665 			**	Set a hacky target so we know to skip to the proper state.
   1666 			*/
   1667 			if (What_Am_I() == RTTI_UNIT && ((UnitClass*)this)->Class->IsToHarvest) {
   1668 				Assign_Mission(MISSION_HARVEST);
   1669 				Assign_Target(As_Target());
   1670 				Assign_Destination(TARGET_NONE);
   1671 			} else {
   1672 				ArchiveTarget = TARGET_NONE;
   1673 				Enter_Idle_Mode();
   1674 			}
   1675 		}
   1676 	}
   1677 	return(TICKS_PER_SECOND/2);
   1678 }
   1679 
   1680 
   1681 /***********************************************************************************************
   1682 z * FootClass::Assign_Destination -- Assigns specified destination to NavCom.                   *
   1683  *                                                                                             *
   1684  *    This routine will assign the specified target to the navigation computer. No legality    *
   1685  *    checks are performed.                                                                    *
   1686  *                                                                                             *
   1687  * INPUT:   target   -- The target value to assign to the navigation computer.                 *
   1688  *                                                                                             *
   1689  * OUTPUT:  none                                                                               *
   1690  *                                                                                             *
   1691  * WARNINGS:   none                                                                            *
   1692  *                                                                                             *
   1693  * HISTORY:                                                                                    *
   1694  *   07/08/1995 JLB : Created.                                                                 *
   1695  *=============================================================================================*/
   1696 void FootClass::Assign_Destination(TARGET target)
   1697 {
   1698 	NavCom = target;
   1699 }
   1700 
   1701 
   1702 /***********************************************************************************************
   1703  * FootClass::Detach_All -- Removes this object from the game system.                          *
   1704  *                                                                                             *
   1705  *    This routine will remove this object from the game system. This routine is called when   *
   1706  *    this object is about to be deleted. All other objects should no longer reference this    *
   1707  *    object in that case.                                                                     *
   1708  *                                                                                             *
   1709  * INPUT:   none                                                                               *
   1710  *                                                                                             *
   1711  * OUTPUT:  none                                                                               *
   1712  *                                                                                             *
   1713  * WARNINGS:   none                                                                            *
   1714  *                                                                                             *
   1715  * HISTORY:                                                                                    *
   1716  *   07/08/1995 JLB : Created.                                                                 *
   1717  *=============================================================================================*/
   1718 void FootClass::Detach_All(bool all)
   1719 {
   1720 	if (Team) Team->Remove(this);
   1721 	Team = NULL;
   1722 
   1723 	TechnoClass::Detach_All(all);
   1724 }
   1725 
   1726 
   1727 /***********************************************************************************************
   1728  * FootClass::Rescue_Mission -- Calls this unit to the rescue.                                 *
   1729  *                                                                                             *
   1730  *    This routine is called when the house determines that it should attack the specified     *
   1731  *    target. This routine will determine if it can attack the target specified and if so,     *
   1732  *    the amount of power it can throw at it. This returned power value is used to allow       *
   1733  *    intelligent distribution of retaliation.                                                 *
   1734  *                                                                                             *
   1735  * INPUT:   target   -- The target that this object just might be assigned to attack and thus  *
   1736  *                      how much power it can bring to bear should be returned.                *
   1737  *                                                                                             *
   1738  * OUTPUT:  Returns with the amount of power that this object can bring to bear against the    *
   1739  *          potential target specified.                                                        *
   1740  *                                                                                             *
   1741  * WARNINGS:   none                                                                            *
   1742  *                                                                                             *
   1743  * HISTORY:                                                                                    *
   1744  *   07/08/1995 JLB : Created.                                                                 *
   1745  *=============================================================================================*/
   1746 int FootClass::Rescue_Mission(TARGET tarcom)
   1747 {
   1748 	/*
   1749 	**	If the target specified is not legal, then it cannot be attacked. Always return
   1750 	**	zero in this case.
   1751 	*/
   1752 	if (!Target_Legal(tarcom)) return(0);
   1753 
   1754 	/*
   1755 	** If the unit is already assigned to destroy the tarcom then we need
   1756 	** to return a negative value which tells the computer to lower the
   1757 	** desired threat rating.
   1758 	*/
   1759 	if (TarCom == tarcom) {
   1760 		return(-Risk());
   1761 	}
   1762 
   1763 	/*
   1764 	** If the unit is currently attacking a target that has a weapon then we
   1765 	** cannot abandon it as it will destroy us if we return to base.
   1766 	*/
   1767 	if (Target_Legal(TarCom)) {
   1768 		TechnoClass * techno = As_Techno(TarCom);
   1769 		if (techno && techno->Techno_Type_Class()->Primary != WEAPON_NONE) {
   1770 			return(0);
   1771 		}
   1772 	}
   1773 
   1774 	/*
   1775 	** If the unit is in a harvest mission or is currently attacking
   1776 	** something, or is not very effective, then it will be of no help
   1777 	** at all.
   1778 	*/
   1779 	if (Team || Mission == MISSION_HARVEST || !Risk()) {
   1780 		return(0);
   1781 	}
   1782 
   1783 	/*
   1784 	** Find the distance to the target modified by the range.  If the
   1785 	** the distance is 0, then things are ok.
   1786 	*/
   1787 	int dist = Distance(tarcom) - Weapon_Range(0);
   1788 	int threat = Risk() * 256;
   1789 	int speed = -1;
   1790 	if (dist > 0) {
   1791 
   1792 		/*
   1793 		** Next we need to figure out how fast the unit moves because this
   1794 		** decreases the distance penalty.
   1795 		*/
   1796 		speed = Max((unsigned)Techno_Type_Class()->MaxSpeed, (unsigned)1);
   1797 
   1798 		int ratio = (speed > 0) ? Max(dist / speed, 1) : 1;
   1799 
   1800 		/*
   1801 		** Finally modify the threat by the distance the unit is away.
   1802 		*/
   1803 		threat = Max(threat/ratio, 1);
   1804 	}
   1805 	return(threat);
   1806 }
   1807 
   1808 
   1809 /***********************************************************************************************
   1810  * FootClass::Death_Announcement -- Announces the death of a unit.                             *
   1811  *                                                                                             *
   1812  *    This routine is called when a unit (infantry, vehicle, or aircraft) is destroyed.        *
   1813  *                                                                                             *
   1814  * INPUT:   source   -- The purpetrator of this death.                                         *
   1815  *                                                                                             *
   1816  * OUTPUT:  none                                                                               *
   1817  *                                                                                             *
   1818  * WARNINGS:   none                                                                            *
   1819  *                                                                                             *
   1820  * HISTORY:                                                                                    *
   1821  *   07/01/1995 JLB : Created.                                                                 *
   1822  *=============================================================================================*/
   1823 void FootClass::Death_Announcement(TechnoClass const * source) const
   1824 {
   1825 	// Changed for multiplayer. ST - 3/13/2019 5:36PM
   1826 	if ((GameToPlay == GAME_GLYPHX_MULTIPLAYER && House->IsHuman) || (GameToPlay != GAME_GLYPHX_MULTIPLAYER && Is_Discovered_By_Player() || Is_Owned_By_Player())) { // ST
   1827 		//if (Is_Discovered_By_Player() || Is_Owned_By_Player()) {
   1828 		//if (IsDiscoveredByPlayer || IsOwnedByPlayer) {
   1829 		if (!source || source->What_Am_I() != RTTI_INFANTRY || *((InfantryClass const *)source) != INFANTRY_RAMBO) {
   1830 			if (What_Am_I() == RTTI_INFANTRY && ((InfantryTypeClass const &)Class_Of()).IsCivilian && !((InfantryClass *)this)->IsTechnician) {
   1831 				// if (Options.IsDeathAnnounce) Speak(VOX_DEAD_CIV); // MBL 02.06.2020
   1832 				if (Options.IsDeathAnnounce) Speak(VOX_DEAD_CIV, House, Center_Coord());
   1833 			} 
   1834 			else {
   1835 				if (House != PlayerPtr && GameToPlay != GAME_NORMAL) {
   1836 					// if (Options.IsDeathAnnounce) Speak(VOX_ENEMY_UNIT); // MBL 02.06.2020
   1837 					if (Options.IsDeathAnnounce) Speak(VOX_ENEMY_UNIT, House);
   1838 				} 
   1839 				else {
   1840 					if (((GameToPlay == GAME_GLYPHX_MULTIPLAYER && House->IsHuman) || House == PlayerPtr) || Options.IsDeathAnnounce) { // ST
   1841 						if (!Options.IsDeathAnnounce) { // MBL 02.06.2020
   1842 							// Speak(VOX_UNIT_LOST);
   1843 							Speak(VOX_UNIT_LOST, House, Center_Coord());
   1844 						} 
   1845 						else {
   1846 							switch (House->ActLike) {
   1847 								case HOUSE_GOOD:
   1848 									// Speak(VOX_DEAD_GDI); // MBL 02.06.2020
   1849 									Speak(VOX_DEAD_GDI, House, Center_Coord());
   1850 									break;
   1851 		
   1852 								case HOUSE_BAD:
   1853 									// Speak(VOX_DEAD_NOD); // MBL 02.06.2020
   1854 									 Speak(VOX_DEAD_NOD, House, Center_Coord());
   1855 									break;
   1856 		
   1857 								default:
   1858 									break;
   1859 							}
   1860 						}
   1861 					}
   1862 				}
   1863 			}
   1864 		}
   1865 	}
   1866 }
   1867 
   1868 
   1869 /***********************************************************************************************
   1870  * FootClass::Greatest_Threat -- Fetches the greatest threat to this object.                   *
   1871  *                                                                                             *
   1872  *    This routine will return with the greatest threat (best target) for this object. For     *
   1873  *    movable ground object, they won't automatically return ANY target if this object is      *
   1874  *    cloaked. Otherwise, cloaking is relatively useless.                                      *
   1875  *                                                                                             *
   1876  * INPUT:   method   -- The request method (bit flags) to use when scanning for a target.      *
   1877  *                                                                                             *
   1878  * OUTPUT:  Returns with the best target to attack. If there is no target that qualifies, then *
   1879  *          TARGET_NONE is returned.                                                           *
   1880  *                                                                                             *
   1881  * WARNINGS:   none                                                                            *
   1882  *                                                                                             *
   1883  * HISTORY:                                                                                    *
   1884  *   07/08/1995 JLB : Created.                                                                 *
   1885  *=============================================================================================*/
   1886 TARGET FootClass::Greatest_Threat(ThreatType method) const
   1887 {
   1888 	/*
   1889 	**	If this object can cloak, then it won't select a target automatically.
   1890 	*/
   1891 	if (House->IsHuman && IsCloakable && Mission == MISSION_GUARD) {
   1892 		return(TARGET_NONE);
   1893 	}
   1894 
   1895 	if (Techno_Type_Class()->Primary != WEAPON_NONE && BulletTypeClass::As_Reference(Weapons[Techno_Type_Class()->Primary].Fires).IsAntiAircraft) {
   1896 		method = method | THREAT_AIR;
   1897 	}
   1898 	if (Techno_Type_Class()->Secondary != WEAPON_NONE && BulletTypeClass::As_Reference(Weapons[Techno_Type_Class()->Secondary].Fires).IsAntiAircraft) {
   1899 		method = method | THREAT_AIR;
   1900 	}
   1901 
   1902 	return(TechnoClass::Greatest_Threat(method|THREAT_GROUND));
   1903 }
   1904 
   1905 
   1906 /***********************************************************************************************
   1907  * FootClass::Detach -- Detaches a target from tracking systems.                               *
   1908  *                                                                                             *
   1909  *    This routine will detach the specified target from the tracking systems of this object.  *
   1910  *    It will be removed from the navigation computer and any queued mission record.           *
   1911  *                                                                                             *
   1912  * INPUT:   target   -- The target to be removed from this object.                             *
   1913  *                                                                                             *
   1914  *          all      -- Is the unit really about to be eliminated? If this is true then even   *
   1915  *                      friendly contact (i.e., radio) must be eliminated.                     *
   1916  *                                                                                             *
   1917  * OUTPUT:  none                                                                               *
   1918  *                                                                                             *
   1919  * WARNINGS:   none                                                                            *
   1920  *                                                                                             *
   1921  * HISTORY:                                                                                    *
   1922  *   07/18/1995 JLB : Created.                                                                 *
   1923  *=============================================================================================*/
   1924 void FootClass::Detach(TARGET target, bool all)
   1925 {
   1926 	TechnoClass::Detach(target, all);
   1927 
   1928 	if (!SpecialFlag) {
   1929 		if (ArchiveTarget == target) {
   1930 			ArchiveTarget = TARGET_NONE;
   1931 		}
   1932 	}
   1933 
   1934 	if (SuspendedNavCom == target) {
   1935 		SuspendedNavCom = TARGET_NONE;
   1936 		SuspendedMission = MISSION_NONE;
   1937 	}
   1938 
   1939 	/*
   1940 	**	If the navigation computer is assigned to the target, then the navigation
   1941 	**	computer must be cleared.
   1942 	*/
   1943 	if (NavCom == target) {
   1944 		NavCom = TARGET_NONE;
   1945 		Path[0] = FACING_NONE;
   1946 		Restore_Mission();
   1947 	}
   1948 
   1949 	/*
   1950 	**	If targeting the specified object and this unit is obviously heading
   1951 	**	toward the target to get within range, then abort the path.
   1952 	*/
   1953 	if (TarCom == target && House->IsHuman) {
   1954 		Path[0] = FACING_NONE;
   1955 	}
   1956 }
   1957 
   1958 
   1959 /***********************************************************************************************
   1960  * FootClass::Offload_Tiberium_Bail -- Fetches the Tiberium to offload per step.               *
   1961  *                                                                                             *
   1962  *    This routine is called when a packet/package/bail of Tiberium needs to be offloaded      *
   1963  *    from the object. This function is overridden for those objects that can contain          *
   1964  *    Tiberium.                                                                                *
   1965  *                                                                                             *
   1966  * INPUT:   none                                                                               *
   1967  *                                                                                             *
   1968  * OUTPUT:  Returns with the number of credits offloaded from the object.                      *
   1969  *                                                                                             *
   1970  * WARNINGS:   This routine must be called multiple times in order to completely offload the   *
   1971  *             Tiberium. When this routine return 0, all Tiberium has been offloaded.          *
   1972  *                                                                                             *
   1973  * HISTORY:                                                                                    *
   1974  *   07/19/1995 JLB : Created.                                                                 *
   1975  *=============================================================================================*/
   1976 int FootClass::Offload_Tiberium_Bail(void)
   1977 {
   1978 	return(0);
   1979 }
   1980 
   1981 
   1982 /***********************************************************************************************
   1983  * FootClass::Can_Enter_Cell -- Checks to see if the object can enter cell specified.          *
   1984  *                                                                                             *
   1985  *    This routine examines the specified cell to see if the object can enter it. This         *
   1986  *    function is to be overridden for objects that could have the possibility of not being    *
   1987  *    allowed to enter the cell. Typical objects at the FootClass level always return          *
   1988  *    MOVE_OK.                                                                                 *
   1989  *                                                                                             *
   1990  * INPUT:   cell     -- The cell to examine.                                                   *
   1991  *                                                                                             *
   1992  *          facing   -- The direction that this cell might be entered from.                    *
   1993  *                                                                                             *
   1994  * OUTPUT:  Returns with the move check result type. This will be MOVE_OK if there is not      *
   1995  *          blockage. There are various other values that represent other blockage types.      *
   1996  *          The value returned will indicatd the most severe reason why entry into the cell    *
   1997  *          is blocked.                                                                        *
   1998  *                                                                                             *
   1999  * WARNINGS:   none                                                                            *
   2000  *                                                                                             *
   2001  * HISTORY:                                                                                    *
   2002  *   07/19/1995 JLB : Created.                                                                 *
   2003  *=============================================================================================*/
   2004 MoveType FootClass::Can_Enter_Cell(CELL , FacingType) const
   2005 {
   2006 	return MOVE_OK;
   2007 }
   2008 
   2009 
   2010 /***********************************************************************************************
   2011  * FootClass::Can_Demolish -- Checks to see if this object can be sold back.                   *
   2012  *                                                                                             *
   2013  *    This routine determines if it is legal to sell the object back. A foot class object can  *
   2014  *    only be sold back if it is sitting on a repair bay.                                      *
   2015  *                                                                                             *
   2016  * INPUT:   none                                                                               *
   2017  *                                                                                             *
   2018  * OUTPUT:  Was the object successfully sold back?                                             *
   2019  *                                                                                             *
   2020  * WARNINGS:   none                                                                            *
   2021  *                                                                                             *
   2022  * HISTORY:                                                                                    *
   2023  *   08/13/1995 JLB : Created.                                                                 *
   2024  *=============================================================================================*/
   2025 bool FootClass::Can_Demolish(void) const
   2026 {
   2027 	switch (What_Am_I()) {
   2028 		case RTTI_UNIT:
   2029 		case RTTI_AIRCRAFT:
   2030 			if (In_Radio_Contact() &&
   2031 				Contact_With_Whom()->What_Am_I() == RTTI_BUILDING &&
   2032 				*((BuildingClass *)Contact_With_Whom()) == STRUCT_REPAIR &&
   2033 				Distance(Contact_With_Whom()) < 0x0080) {
   2034 
   2035 				return(true);
   2036 			}
   2037 			break;
   2038 
   2039 		default:
   2040 			break;
   2041 	}
   2042 	return(TechnoClass::Can_Demolish());
   2043 }
   2044 
   2045 
   2046 /***********************************************************************************************
   2047  * FootClass::Sell_Back -- Causes this object to be sold back.                                 *
   2048  *                                                                                             *
   2049  *    When an object is sold back, a certain amount of money is refunded to the owner and then *
   2050  *    the object is removed from the game system.                                              *
   2051  *                                                                                             *
   2052  * INPUT:   control  -- The action to perform. The only supported action is "1", which means   *
   2053  *                      to sell back.                                                          *
   2054  *                                                                                             *
   2055  * OUTPUT:  none                                                                               *
   2056  *                                                                                             *
   2057  * WARNINGS:   none                                                                            *
   2058  *                                                                                             *
   2059  * HISTORY:                                                                                    *
   2060  *   08/13/1995 JLB : Created.                                                                 *
   2061  *=============================================================================================*/
   2062 void FootClass::Sell_Back(int control)
   2063 {
   2064 	if (control != 0) {
   2065 		if (House == PlayerPtr) {
   2066 			Sound_Effect(VOC_CASHTURN);
   2067 		}
   2068 		House->Refund_Money(Refund_Amount());
   2069 		Stun();
   2070 		Limbo();
   2071 		Delete_This();
   2072 	}
   2073 }
   2074 
   2075 
   2076 /***********************************************************************************************
   2077  * FootClass::Likely_Coord -- Fetches the coordinate the object will be at shortly.            *
   2078  *                                                                                             *
   2079  *    This routine comes in handy when determining where a travelling object will be at        *
   2080  *    when considering the amount of time it would take for a normal unit to travel one cell.  *
   2081  *    Using this information, an intelligent "approach target" logic can be employed.          *
   2082  *                                                                                             *
   2083  * INPUT:   none                                                                               *
   2084  *                                                                                             *
   2085  * OUTPUT:  Returns with the coordinate the object is at or soon will be.                      *
   2086  *                                                                                             *
   2087  * WARNINGS:   none                                                                            *
   2088  *                                                                                             *
   2089  * HISTORY:                                                                                    *
   2090  *   08/13/1995 JLB : Created.                                                                 *
   2091  *=============================================================================================*/
   2092 COORDINATE FootClass::Likely_Coord(void) const
   2093 {
   2094 	if (Head_To_Coord()) {
   2095 		return(Head_To_Coord());
   2096 	}
   2097 	return(Target_Coord());
   2098 }