CnC_Remastered_Collection

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

AIRCRAFT.CPP (159627B)


      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\aircraft.cpv   2.12   19 Jun 1995 09:27:22   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 : AIRCRAFT.CPP                                                 *
     24  *                                                                                             *
     25  *                   Programmer : Joe L. Bostic                                                *
     26  *                                                                                             *
     27  *                   Start Date : July 22, 1994                                                *
     28  *                                                                                             *
     29  *                  Last Update : August 13, 1995 [JLB]                                        *
     30  *                                                                                             *
     31  *---------------------------------------------------------------------------------------------*
     32  * Functions:                                                                                  *
     33  *   AircraftClass::AI -- Processes the normal non-graphic AI for the aircraft.                *
     34  *   AircraftClass::Active_Click_With -- Handles clicking over specified cell.                 *
     35  *   AircraftClass::Active_Click_With -- Handles clicking over specified object.               *
     36  *   AircraftClass::AircraftClass -- The constructor for aircraft objects.                     *
     37  *   AircraftClass::As_Target -- Returns aircraft as a target number.                          *
     38  *   AircraftClass::Can_Enter_Cell -- Determines if the aircraft can land at this location.    *
     39  *   AircraftClass::Cell_Seems_Ok -- Checks to see if a cell is good to enter.                 *
     40  *   AircraftClass::Desired_Load_Dir -- Determines where passengers should line up.            *
     41  *   AircraftClass::Draw_It -- Renders an aircraft object at the location specified.           *
     42  *   AircraftClass::Enter_Idle_Mode -- Gives the aircraft an appropriate mission.              *
     43  *   AircraftClass::Exit_Object -- Unloads passenger from aircraft.                            *
     44  *   AircraftClass::Fire_At -- Handles firing a projectile from an aircraft.                   *
     45  *   AircraftClass::Fire_Coord -- Calculates the point of origin for a bullet.                 *
     46  *   AircraftClass::Fire_Direction -- Determines the direction of fire.                        *
     47  *   AircraftClass::Good_Fire_Location -- Searches for and finds a good spot to fire from.     *
     48  *   AircraftClass::Good_LZ -- Locates a good spot ot land.                                    *
     49  *   AircraftClass::In_Which_Layer -- Determine which render layer the aircraft lies.          *
     50  *   AircraftClass::Init -- Initialize the aircraft system to an empty state.                  *
     51  *   AircraftClass::Is_LZ_Clear -- Determines if landing zone is free for landing.             *
     52  *   AircraftClass::Mark -- Flags cells under the aircraft so that they will be redrawn.       *
     53  *   AircraftClass::Mission_Attack -- Handles the attack mission for aircraft.                 *
     54  *   AircraftClass::Mission_Enter -- Control aircraft to fly to the helipad or repair center.  *
     55  *   AircraftClass::Mission_Guard -- Handles aircraft in guard mode.                           *
     56  *   AircraftClass::Mission_Guard_Area -- Handles the aircraft guard area logic.               *
     57  *   AircraftClass::Mission_Hunt -- Maintains hunt AI for the aircraft.                        *
     58  *   AircraftClass::Mission_Move -- Handles movement mission.                                  *
     59  *   AircraftClass::Mission_Retreat -- Handles the aircraft logic for leaving the battlefield. *
     60  *   AircraftClass::Mission_Unload -- Handles unloading cargo.                                 *
     61  *   AircraftClass::New_LZ -- Find a good landing zone.                                        *
     62  *   AircraftClass::Overlap_List -- Returns with list of cells the aircraft overlaps.          *
     63  *   AircraftClass::Pip_Count -- Returns the number of "objects" in aircraft.                  *
     64  *   AircraftClass::Player_Assign_Mission -- Handles player input to assign a mission.         *
     65  *   AircraftClass::Pose_Dir -- Fetches the natural landing facing.                            *
     66  *   AircraftClass::Process_Fly_To -- Handles state machine for flying to destination.         *
     67  *   AircraftClass::Process_Landing -- Landing process state machine handler.                  *
     68  *   AircraftClass::Process_Take_Off -- State machine support for taking off.                  *
     69  *   AircraftClass::Read_INI -- Reads aircraft object data from an INI file.                   *
     70  *   AircraftClass::Rearm_Delay -- Returns the delay between shots for this aircraft.          *
     71  *   AircraftClass::Receive_Message -- Handles receipt of radio messages.                      *
     72  *   AircraftClass::Response_Attack -- Gives audio response to attack order.                   *
     73  *   AircraftClass::Response_Move -- Gives audio response to move request.                     *
     74  *   AircraftClass::Response_Select -- Gives audio response when selected.                     *
     75  *   AircraftClass::Scatter -- Causes the aircraft to move away a bit.                         *
     76  *   AircraftClass::Set_Speed -- Sets the speed for the aircraft.                              *
     77  *   AircraftClass::Sort_Y -- Figures the sorting coordinate.                                  *
     78  *   AircraftClass::Take_Damage -- Applies damage to the aircraft.                             *
     79  *   AircraftClass::Threat_Range -- Returns with a range to scan for targets.                  *
     80  *   AircraftClass::Unlimbo -- Removes an aircraft from the limbo state.                       *
     81  *   AircraftClass::What_Action -- Determines what action to perform.                          *
     82  *   AircraftClass::What_Action -- Determines what action to perform.                          *
     83  *   AircraftClass::Write_INI -- Writes the current aircraft objects to an INI file.           *
     84  *   AircraftClass::operator delete -- Deletes the aircraft object.                            *
     85  *   AircraftClass::operator new -- Allocates a new aircraft object from the pool              *
     86  *   AircraftClass::~AircraftClass -- Destructor for aircraft object.                          *
     87  *   AircraftClass::Validate -- validates aircraft pointer												  *
     88  * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     89 
     90 #include	"function.h"
     91 
     92 
     93 /*
     94 ** This contains the value of the Virtual Function Table Pointer
     95 */
     96 void * AircraftClass::VTable;
     97 
     98 
     99 /***********************************************************************************************
    100  * AircraftClass::Validate -- validates aircraft pointer													  *
    101  *                                                                                             *
    102  * INPUT:                                                                                      *
    103  *		none.																												  *
    104  *                                                                                             *
    105  * OUTPUT:                                                                                     *
    106  *		1 = ok, 0 = error																								  *
    107  *                                                                                             *
    108  * WARNINGS:                                                                                   *
    109  *		none.																												  *
    110  *                                                                                             *
    111  * HISTORY:                                                                                    *
    112  *   08/09/1995 BRR : Created.                                                                 *
    113  *=============================================================================================*/
    114 #ifdef CHEAT_KEYS
    115 int AircraftClass::Validate(void) const
    116 {
    117 	int num;
    118 
    119 	num = Aircraft.ID(this);
    120 	if (num < 0 || num >= AIRCRAFT_MAX) {
    121 		Validate_Error("AIRCRAFT");
    122 		return (0);
    123 	}
    124 	else
    125 		return (1);
    126 }
    127 #else
    128 #define	Validate()
    129 #endif
    130 
    131 
    132 /***********************************************************************************************
    133  * AircraftClass::As_Target -- Returns aircraft as a target number.                            *
    134  *                                                                                             *
    135  *    This routine will convert the aircraft into a target number. This target number can      *
    136  *    then be assigned to a targeting or navigation computer.                                  *
    137  *                                                                                             *
    138  * INPUT:   none                                                                               *
    139  *                                                                                             *
    140  * OUTPUT:  Returns the aircraft as a target number.                                           *
    141  *                                                                                             *
    142  * WARNINGS:   none                                                                            *
    143  *                                                                                             *
    144  * HISTORY:                                                                                    *
    145  *   09/24/1994 JLB : Created.                                                                 *
    146  *=============================================================================================*/
    147 TARGET AircraftClass::As_Target(void) const
    148 {
    149 	Validate();
    150 	return(Build_Target(KIND_AIRCRAFT, Aircraft.ID(this)));
    151 }
    152 
    153 
    154 /***********************************************************************************************
    155  * AircraftClass::operator new -- Allocates a new aircraft object from the pool                *
    156  *                                                                                             *
    157  *    This routine will allocate an aircraft object from the free aircraft object pool. If     *
    158  *    there are no free object available, then this routine will fail (return NULL).           *
    159  *                                                                                             *
    160  * INPUT:   none                                                                               *
    161  *                                                                                             *
    162  * OUTPUT:  Returns with a pointer to the allocate aircraft object or NULL if none were        *
    163  *          available.                                                                         *
    164  *                                                                                             *
    165  * WARNINGS:   none                                                                            *
    166  *                                                                                             *
    167  * HISTORY:                                                                                    *
    168  *   07/26/1994 JLB : Created.                                                                 *
    169  *=============================================================================================*/
    170 void * AircraftClass::operator new(size_t)
    171 {
    172 	void * ptr = Aircraft.Allocate();
    173 	if (ptr) {
    174 		((AircraftClass *)ptr)->Set_Active();
    175 	}
    176 	return(ptr);
    177 }
    178 
    179 
    180 /***********************************************************************************************
    181  * AircraftClass::operator delete -- Deletes the aircraft object.                              *
    182  *                                                                                             *
    183  *    This routine will return the aircraft object back to the free aircraft object pool.      *
    184  *                                                                                             *
    185  * INPUT:   ptr   -- Pointer to the aircraft object to delete.                                 *
    186  *                                                                                             *
    187  * OUTPUT:  none                                                                               *
    188  *                                                                                             *
    189  * WARNINGS:   none                                                                            *
    190  *                                                                                             *
    191  * HISTORY:                                                                                    *
    192  *   07/26/1994 JLB : Created.                                                                 *
    193  *=============================================================================================*/
    194 void AircraftClass::operator delete(void *ptr)
    195 {
    196 	if (ptr) {
    197 		((AircraftClass *)ptr)->IsActive = false;
    198 	}
    199 	Aircraft.Free((AircraftClass *)ptr);
    200 
    201 	//Map.Validate();
    202 }
    203 
    204 
    205 /***********************************************************************************************
    206  * AircraftClass::AircraftClass -- The constructor for aircraft objects.                       *
    207  *                                                                                             *
    208  *    This routine is the constructor for aircraft objects. An aircraft object can be          *
    209  *    created and possibly placed into the game system by this routine.                        *
    210  *                                                                                             *
    211  * INPUT:   classid  -- The type of aircraft to create.                                        *
    212  *                                                                                             *
    213  *          house    -- The owner of this aircraft.                                            *
    214  *                                                                                             *
    215  * OUTPUT:  none                                                                               *
    216  *                                                                                             *
    217  * WARNINGS:   none                                                                            *
    218  *                                                                                             *
    219  * HISTORY:                                                                                    *
    220  *   07/26/1994 JLB : Created.                                                                 *
    221  *=============================================================================================*/
    222 AircraftClass::AircraftClass(AircraftType classid, HousesType house) :
    223 	Class(&AircraftTypeClass::As_Reference(classid)),
    224 	FootClass(house)
    225 {
    226 	/*
    227 	**	For two shooters, clear out the second shot flag -- it will be set the first time
    228 	**	the object fires. For non two shooters, set the flag since it will never be cleared
    229 	**	and the second shot flag tells the system that normal rearm times apply -- this is
    230 	**	what is desired for non two shooters.
    231 	*/
    232 	if (Class->IsTwoShooter) {
    233 		IsSecondShot = false;
    234 	} else {
    235 		IsSecondShot = true;
    236 	}
    237 	Ammo = Class->MaxAmmo;
    238 	AttacksRemaining = 3;
    239 	Altitude = FLIGHT_LEVEL;
    240 	IsLanding = false;
    241 	IsTakingOff = false;
    242 	IsHovering = false;
    243 	IsHoming = false;
    244 	Strength = Class->MaxStrength;
    245 	NavCom = TARGET_NONE;
    246 	SecondaryFacing = PrimaryFacing;
    247 	Jitter = 0;
    248 	ReinforcementStart = -1;
    249 
    250 	/*
    251 	** Keep count of the number of units created. Dont track cargo planes as they are created
    252 	** automatically, not bought.
    253 	*/
    254 	if (classid != AIRCRAFT_CARGO && GameToPlay == GAME_INTERNET){
    255 		House->AircraftTotals->Increment_Unit_Total((int)classid);
    256 	}
    257 
    258 #ifdef USE_RA_AI
    259 	//
    260 	// Added for RA AI in TD. ST - 7/26/2019 9:12AM
    261 	//
    262 	House->Tracking_Add(this);
    263 #endif
    264 }
    265 
    266 
    267 /***********************************************************************************************
    268  * AircraftClass::Unlimbo -- Removes an aircraft from the limbo state.                         *
    269  *                                                                                             *
    270  *    This routine is used to transition the aircraft from the limbo to the non limbo state.   *
    271  *    It occurs when the aircraft is placed on the map for whatever reason. When it is         *
    272  *    unlimboed, only then will normal game processing recognize it.                           *
    273  *                                                                                             *
    274  * INPUT:   coord -- The coordinate that the aircraft should appear at.                        *
    275  *                                                                                             *
    276  *          dir   -- The direction it should start facing.                                     *
    277  *                                                                                             *
    278  *				strength (optional) -- sets initial strength													  *
    279  *                                                                                             *
    280  *				mission (optional) -- sets initial mission													  *
    281  *                                                                                             *
    282  * OUTPUT:  bool; Was the aircraft unlimboed successfully?                                     *
    283  *                                                                                             *
    284  * WARNINGS:   none                                                                            *
    285  *                                                                                             *
    286  * HISTORY:                                                                                    *
    287  *   07/26/1994 JLB : Created.                                                                 *
    288  *=============================================================================================*/
    289 bool AircraftClass::Unlimbo(COORDINATE coord, DirType dir)
    290 {
    291 	Validate();
    292 	if (FootClass::Unlimbo(coord, dir)) {
    293 
    294 		/*
    295 		**	Ensure that the owning house knows about the
    296 		**	new object.
    297 		*/
    298 		House->AScan |= (1L << Class->Type);
    299 		House->ActiveAScan |= (1L << Class->Type);
    300 
    301 		/*
    302 		**	Forces the body of the helicopter to face the correct direction.
    303 		*/
    304 		SecondaryFacing = dir;
    305 
    306 		/*
    307 		**	Start rotor animation.
    308 		*/
    309 		Set_Rate(1);
    310 		Set_Stage(0);
    311 
    312 		/*
    313 		**	Presume it starts in flight?
    314 		*/
    315 		if (Altitude == FLIGHT_LEVEL) {
    316 			Set_Speed(0xFF);
    317 		} else {
    318 			Set_Speed(0);
    319 		}
    320 		return(true);
    321 	}
    322 	return(false);
    323 }
    324 
    325 
    326 /***********************************************************************************************
    327  * AircraftClass::Draw_It -- Renders an aircraft object at the location specified.             *
    328  *                                                                                             *
    329  *    This routine is used to display the aircraft object at the coordinates specified.        *
    330  *    The tactical map display uses this routine for all aircraft rendering.                   *
    331  *                                                                                             *
    332  * INPUT:   x,y      -- The coordinates to render the aircraft at.                             *
    333  *                                                                                             *
    334  *          window   -- The window that the coordinates are based upon.                        *
    335  *                                                                                             *
    336  * OUTPUT:  none                                                                               *
    337  *                                                                                             *
    338  * WARNINGS:   none                                                                            *
    339  *                                                                                             *
    340  * HISTORY:                                                                                    *
    341  *   07/26/1994 JLB : Created.                                                                 *
    342  *=============================================================================================*/
    343 void AircraftClass::Draw_It(int x, int y, WindowNumberType window)
    344 {
    345 	Validate();
    346 	void const * shapefile;		// Working shape file pointer.
    347 	int shapenum = 0;
    348 	int facing = Facing_To_32(SecondaryFacing);
    349 
    350 	/*
    351 	**	Don't draw Cargo aircraft that are delayed.
    352 	*/
    353 	if (Special.ModernBalance) {
    354 		if (*this == AIRCRAFT_CARGO && !Map.In_Radar(Coord_Cell(Coord)) && ReinforcementStart > Frame) {
    355 			return;
    356 		}
    357 	}
    358 
    359 	/*
    360 	**	Verify the legality of the unit class.
    361 	*/
    362 	shapefile = Class->Get_Image_Data();
    363 	if (!shapefile) return;
    364 	shapenum = UnitClass::BodyShape[facing];
    365 
    366 	/*
    367 	**	The orca attack helicopter uses a special shape set when it is travelling
    368 	**	forward above a certain speed.
    369 	*/
    370 	if (*this == AIRCRAFT_ORCA && Get_Speed() >= MPH_MEDIUM_FAST) {
    371 		shapenum += 32;
    372 	}
    373 
    374 	/*
    375 	**	If there is a door on this aircraft (Chinook), then adjust the
    376 	**	shape number to match the door open state.
    377 	*/
    378 	if (!Is_Door_Closed()) {
    379 		shapenum = 32 + Door_Stage();
    380 	}
    381 
    382 	/*
    383 	**	Helicopters that are flying have a "bobbing" effect.
    384 	*/
    385 	int jitter = 0;
    386 	if (Altitude == FLIGHT_LEVEL && !Class->IsFixedWing) {
    387 		Jitter++;
    388 
    389 		static int _jitter[] = {0,0,0,0,1,1,1,0,0,0,0,0,-1,-1,-1,0};
    390 		jitter = _jitter[Jitter % 16];
    391 	}
    392 
    393 	// Virtual window needs to draw the body first so it's considered the primary object and the shadow is a sub-object
    394 	if (window == WINDOW_VIRTUAL) {
    395 		/*
    396 		**	Draw the root body of the unit.
    397 		*/
    398 		Techno_Draw_Object(shapefile, shapenum, x, (y - Altitude) + jitter, window);
    399 
    400 		/*
    401 		**	Special manual shadow draw code.
    402 		*/
    403 		if (Visual_Character() <= VISUAL_DARKEN) {
    404 			CC_Draw_Shape(this, shapefile, shapenum, x + 1, y + 2, window, SHAPE_PREDATOR | SHAPE_CENTER | SHAPE_WIN_REL | SHAPE_FADING, Map.FadingShade, NULL);
    405 		}
    406 	} else {
    407 		/*
    408 		**	Special manual shadow draw code.
    409 		*/
    410 		if (Visual_Character() <= VISUAL_DARKEN) {
    411 			CC_Draw_Shape(this, shapefile, shapenum, x+1, y+2, window, SHAPE_PREDATOR|SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING, Map.FadingShade, NULL);
    412 		}
    413 
    414 		/*
    415 		**	Draw the root body of the unit.
    416 		*/
    417 		Techno_Draw_Object(shapefile, shapenum, x, (y-Altitude)+jitter, window);
    418 	}
    419 
    420 	/*
    421 	**	Draw rotor effects. The rotor art can be either generic or custom. Custom rotor
    422 	**	art has a different rotor set for each facing. Rotor shapes occur after the first
    423 	**	32 shapes of the helicopter body.
    424 	*/
    425 	if (Class->IsRotorEquipped) {
    426 		ShapeFlags_Type flags = SHAPE_CENTER|SHAPE_WIN_REL;
    427 
    428 		/*
    429 		**	The rotor shape number depends on whether the helicopter is idling
    430 		**	or not. A landed helicopter uses slow moving "idling" blades.
    431 		*/
    432 		if (Altitude == 0) {
    433 			shapenum = (Fetch_Stage()%8)+4;
    434 			flags = flags | SHAPE_GHOST;
    435 		} else {
    436 			shapenum = Fetch_Stage()%4;
    437 			flags = flags | SHAPE_FADING|SHAPE_PREDATOR;
    438 		}
    439 
    440 		if (*this == AIRCRAFT_TRANSPORT) {
    441 			int _stretch[FACING_COUNT] = {8, 9, 10, 9, 8, 9, 10, 9};
    442 
    443 			/*
    444 			**	Dual rotors offset along flight axis.
    445 			*/
    446 			short xx = x;
    447 			short yy = y-Altitude;
    448 			FacingType face = Dir_Facing(SecondaryFacing);
    449 			Move_Point(xx, yy, SecondaryFacing.Current(), _stretch[face]);
    450 			//CC_Draw_Shape(Class->RRotorData, shapenum, xx, yy-2, window, flags, NULL, Map.UnitShadow);		// 6/18/2019 - ST
    451 			CC_Draw_Shape(this, "RROTOR", Class->RRotorData, shapenum, xx, yy - 2, window, flags, NULL, Map.UnitShadow);
    452 
    453 			Move_Point(xx, yy, SecondaryFacing.Current()+DIR_S, _stretch[face]*2);
    454 			//CC_Draw_Shape(this, Class->LRotorData, shapenum, xx, yy-2, window, flags, NULL, Map.UnitShadow);			// 6/18/2019 - ST
    455 			CC_Draw_Shape(this, "LROTOR", Class->LRotorData, shapenum, xx, yy - 2, window, flags, NULL, Map.UnitShadow);
    456 
    457 		} else {
    458 
    459 			/*
    460 			**	Single rotor centered about shape.
    461 			*/
    462 			//CC_Draw_Shape(this, Class->RRotorData, shapenum, x, (y-Altitude)-2, window, flags, NULL, Map.UnitShadow);	// 6/18/2019 - ST
    463 			CC_Draw_Shape(this, "RROTOR", Class->RRotorData, shapenum, x, (y - Altitude) - 2, window, flags, NULL, Map.UnitShadow);
    464 		}
    465 	}
    466 
    467 	FootClass::Draw_It(x, y-Altitude, window);
    468 }
    469 
    470 
    471 /***********************************************************************************************
    472  * AircraftClass::Read_INI -- Reads aircraft object data from an INI file.                     *
    473  *                                                                                             *
    474  *    This routine is used to read the aircraft object data from the INI file buffer           *
    475  *    specified. This is used by the scenario loader code to interpret the INI file and        *
    476  *    create the specified objects therein.                                                    *
    477  *                                                                                             *
    478  * INPUT:   buffer   -- Pointer to the INI buffer.                                             *
    479  *                                                                                             *
    480  * OUTPUT:  none                                                                               *
    481  *                                                                                             *
    482  * WARNINGS:   none                                                                            *
    483  *                                                                                             *
    484  * HISTORY:                                                                                    *
    485  *   07/26/1994 JLB : Created.                                                                 *
    486  *=============================================================================================*/
    487 void AircraftClass::Read_INI(char *buffer)
    488 {
    489 	AircraftClass	*air;			// Working unit pointer.
    490 	char				*tbuffer;		// Accumulation buffer of unit IDs.
    491 	HousesType		inhouse;			// Unit house.
    492 	AircraftType	classid;			// Unit class.
    493 	int				len;				// Length of data in buffer.
    494 	char				buf[128];
    495 
    496 	len = strlen(buffer) + 2;
    497 	tbuffer = buffer + len;
    498 
    499 	WWGetPrivateProfileString(INI_Name(), NULL, NULL, tbuffer, ShapeBufferSize-len, buffer);
    500 	while (*tbuffer != '\0') {
    501 
    502 		WWGetPrivateProfileString(INI_Name(), tbuffer, NULL, buf, sizeof(buf)-1, buffer);
    503 		inhouse = HouseTypeClass::From_Name(strtok(buf, ","));
    504 		if (inhouse != HOUSE_NONE) {
    505 			classid = AircraftTypeClass::From_Name(strtok(NULL, ","));
    506 
    507 			if (classid != AIRCRAFT_NONE) {
    508 
    509 				if (HouseClass::As_Pointer(inhouse) != NULL) {
    510 					air = new AircraftClass(classid, inhouse);
    511 					if (air) {
    512 						COORDINATE	coord;
    513 						int	strength;
    514 						DirType dir;
    515 
    516 						/*
    517 						**	Read the raw data.
    518 						*/
    519 						strength = atoi(strtok(NULL, ","));
    520 						coord = Cell_Coord((CELL)atoi(strtok(NULL, ",")));
    521 						dir = (DirType)atoi(strtok(NULL, ","));
    522 
    523 						if (!Map.In_Radar(Coord_Cell(coord))) {
    524 							delete air;
    525 						} else {
    526 
    527 							air->Strength = Fixed_To_Cardinal(air->Class->MaxStrength, strength);
    528 							if (air->Unlimbo(coord, dir)) {
    529 								air->Assign_Mission(AircraftClass::Mission_From_Name(strtok(NULL, ",\n\r")));
    530 							} else {
    531 								delete air;
    532 							}
    533 						}
    534 					}
    535 				}
    536 			}
    537 		}
    538 		tbuffer += strlen(tbuffer)+1;
    539 	}
    540 }
    541 
    542 
    543 /***********************************************************************************************
    544  * AircraftClass::Write_INI -- Writes the current aircraft objects to an INI file.             *
    545  *                                                                                             *
    546  *    This routine is used to output the current list of aircraft objects to the INI file      *
    547  *    buffer specified. It is typically used by the scenario editor.                           *
    548  *                                                                                             *
    549  * INPUT:   buffer   -- Pointer to the buffer that holds the INI file data.                    *
    550  *                                                                                             *
    551  * OUTPUT:  none                                                                               *
    552  *                                                                                             *
    553  * WARNINGS:   none                                                                            *
    554  *                                                                                             *
    555  * HISTORY:                                                                                    *
    556  *   07/26/1994 JLB : Created.                                                                 *
    557  *=============================================================================================*/
    558 void AircraftClass::Write_INI(char *buffer)
    559 {
    560 	int	index;
    561 	char	uname[10];
    562 	char	buf[128];
    563 	char	*tbuffer;		// Accumulation buffer of unit IDs.
    564 
    565 	/*
    566 	**	First, clear out all existing unit data from the ini file.
    567 	*/
    568 	tbuffer = buffer + strlen(buffer) + 2;
    569 	WWGetPrivateProfileString(INI_Name(), NULL, NULL, tbuffer, ShapeBufferSize-strlen(buffer), buffer);
    570 	while (*tbuffer != '\0') {
    571 		WWWritePrivateProfileString(INI_Name(), tbuffer, NULL, buffer);
    572 		tbuffer += strlen(tbuffer)+1;
    573 	}
    574 
    575 	/*
    576 	**	Write the unit data out.
    577 	*/
    578 	for (index = 0; index < Aircraft.Count(); index++) {
    579 		AircraftClass * unit;
    580 
    581 		unit = Aircraft.Ptr(index);
    582 		if (!unit->IsInLimbo) {
    583 
    584 			sprintf(uname, "%03d", index);
    585 			sprintf(buf, "%s,%s,%d,%u,%d,%s",
    586 				unit->House->Class->IniName,
    587 				unit->Class->IniName,
    588 				unit->Health_Ratio(),
    589 				Coord_Cell(unit->Coord),
    590 				unit->PrimaryFacing.Current(),
    591 				MissionClass::Mission_Name(unit->Mission)
    592 				);
    593 			WWWritePrivateProfileString(INI_Name(), uname, buf, buffer);
    594 		}
    595 	}
    596 }
    597 
    598 
    599 /***********************************************************************************************
    600  * AircraftClass::Mission_Hunt -- Maintains hunt AI for the aircraft.                          *
    601  *                                                                                             *
    602  *    Hunt AI consists of finding a target and attacking it. If there is no target assigned    *
    603  *    and this unit doesn't automatically hunt for more targets, then it will change           *
    604  *    mission to a more passive (land and await further orders) type.                          *
    605  *                                                                                             *
    606  * INPUT:   none                                                                               *
    607  *                                                                                             *
    608  * OUTPUT:  Returns with the number of ticks before calling this routine again.                *
    609  *                                                                                             *
    610  * WARNINGS:   none                                                                            *
    611  *                                                                                             *
    612  * HISTORY:                                                                                    *
    613  *   07/26/1994 JLB : Created.                                                                 *
    614  *=============================================================================================*/
    615 int AircraftClass::Mission_Hunt(void)
    616 {
    617 	Validate();
    618 	if (Class->IsFixedWing) {
    619 		if (Class->Primary == WEAPON_NONE && Class->Secondary == WEAPON_NONE) {
    620 			Assign_Mission(MISSION_RETREAT);
    621 			return(1);
    622 		}
    623 		enum {
    624 			LOOK_FOR_TARGET,
    625 			FLY_TO_TARGET,
    626 			DROP_BOMBS
    627 		};
    628 		switch (Status) {
    629 
    630 			/*
    631 			**	Acquiring target stage.
    632 			*/
    633 			case LOOK_FOR_TARGET:
    634 				if (Target_Legal(TarCom)) {
    635 					Status = FLY_TO_TARGET;
    636 					return(1);
    637 				} else {
    638 					Assign_Target(Greatest_Threat(THREAT_NORMAL));
    639 
    640 					/*
    641 					**	If there is no target, then this aircraft should just do its normal thing.
    642 					*/
    643 					if (!Target_Legal(TarCom)) {
    644 						Enter_Idle_Mode();
    645 					}
    646 				}
    647 				break;
    648 
    649 			/*
    650 			**	Homing in on target stage.
    651 			*/
    652 			case FLY_TO_TARGET:
    653 				if (Target_Legal(TarCom)) {
    654 					IsHoming = true;
    655 					if (!PrimaryFacing.Is_Rotating()) {
    656 						PrimaryFacing.Set_Desired(Direction(TarCom));
    657 					}
    658 					if (Distance(TarCom) < 0x0380) {
    659 						IsHoming = false;
    660 						Status = DROP_BOMBS;
    661 						return(1);
    662 					}
    663 				} else {
    664 					Status = LOOK_FOR_TARGET;
    665 				}
    666 				break;
    667 
    668 			/*
    669 			**	Dropping a stream of bombs phase.
    670 			*/
    671 			case DROP_BOMBS:
    672 				if (!Ammo) {
    673 					AttacksRemaining--;
    674 					if (!AttacksRemaining) {
    675 						Assign_Mission(MISSION_RETREAT);
    676 						Commence();
    677 					} else {
    678 						Ammo = Class->MaxAmmo;
    679 						Status = LOOK_FOR_TARGET;
    680 					}
    681 				}
    682 				if (!Target_Legal(TarCom)) {
    683 					Status = LOOK_FOR_TARGET;
    684 				} else {
    685 					Fire_At(TarCom, 0);
    686 					Map[::As_Cell(TarCom)].Incoming(Coord, true);
    687 					return(5);
    688 				}
    689 				break;
    690 		}
    691 	} else {
    692 		if (!Ammo) {
    693 			Enter_Idle_Mode();
    694 		} else {
    695 			Assign_Mission(MISSION_ATTACK);
    696 			return(1);
    697 //			return(FootClass::Mission_Hunt());
    698 		}
    699 	}
    700 	return(TICKS_PER_SECOND);
    701 }
    702 
    703 
    704 /***********************************************************************************************
    705  * AircraftClass::AI -- Processes the normal non-graphic AI for the aircraft.                  *
    706  *                                                                                             *
    707  *    This handles the non-graphic AI processing for the aircraft. This usually entails        *
    708  *    maintenance and other AI functions.                                                      *
    709  *                                                                                             *
    710  * INPUT:   none                                                                               *
    711  *                                                                                             *
    712  * OUTPUT:  none                                                                               *
    713  *                                                                                             *
    714  * WARNINGS:   none                                                                            *
    715  *                                                                                             *
    716  * HISTORY:                                                                                    *
    717  *   07/26/1994 JLB : Created.                                                                 *
    718  *=============================================================================================*/
    719 void AircraftClass::AI(void)
    720 {
    721 	Validate();
    722 	/*
    723 	**	A Mission change can always occur if the aircraft is landed or flying.
    724 	*/
    725 	if (!IsLanding && !IsTakingOff) {
    726 		Commence();
    727 	}
    728 
    729 	FootClass::AI();
    730 
    731 	/*
    732 	**	A Mission change can always occur if the aircraft is landed or flying.
    733 	*/
    734 	if (!IsLanding && !IsTakingOff) {
    735 		Commence();
    736 	}
    737 
    738 	/*
    739 	**	Handle any body rotation at this time. Body rotation can occur even if the
    740 	**	flying object is not actually moving.
    741 	*/
    742 	if (PrimaryFacing.Is_Rotating()) {
    743 		if (PrimaryFacing.Rotation_Adjust(Class->ROT)) {
    744 			Mark();
    745 		}
    746 	}
    747 	if (Class->IsFixedWing) {
    748 		SecondaryFacing = PrimaryFacing;
    749 	}
    750 	if (SecondaryFacing.Is_Rotating()) {
    751 		if (SecondaryFacing.Rotation_Adjust(Class->ROT)) {
    752 			Mark();
    753 		}
    754 	}
    755 
    756 	/*
    757 	**	Handle reinforcement delay.
    758 	*/
    759 	bool do_physics = true;
    760 	if (Special.ModernBalance) {
    761 		if (*this == AIRCRAFT_CARGO && !Map.In_Radar(Coord_Cell(Coord)) && ReinforcementStart > Frame) {
    762 			do_physics = false;
    763 		}
    764 	}
    765 	if (do_physics && Physics(Coord, PrimaryFacing) != RESULT_NONE) {
    766 		Mark();
    767 	}
    768 
    769 	/*
    770 	**	Perform sighting every so often as controlled by the sight timer.
    771 	*/
    772 	//if (IsOwnedByPlayer && Class->SightRange && SightTimer.Expired()) {		// Changed for multiple player mapping
    773 	if (House->IsHuman && Class->SightRange && SightTimer.Expired()) {
    774 		Map.Sight_From(House, Coord_Cell(Coord), Class->SightRange, false);
    775 		SightTimer = TICKS_PER_SECOND;
    776 	}
    777 
    778 	/*
    779 	**	Handle landing and taking off logic. Helicopters are prime users of this technique. The
    780 	**	aircraft will either gain or lose altitude as appropriate. As the aircraft transitions
    781 	**	between flying level and ground level, it will be moved into the appropriate render
    782 	**	layer.
    783 	*/
    784 	if (Is_Door_Closed() && (IsLanding || IsTakingOff)) {
    785 		Mark();
    786 		LayerType layer = In_Which_Layer();
    787 
    788 		if (IsLanding) {
    789 			if (Altitude) Altitude--;
    790 			if (!Altitude) {
    791 				IsLanding = false;
    792 				Set_Speed(0);
    793 				if (Target_Legal(NavCom) && As_Techno(NavCom) == Contact_With_Whom()) {
    794 					if (In_Radio_Contact() && Transmit_Message(RADIO_IM_IN) != RADIO_ROGER) {
    795 						Scatter(0, true);
    796 					}
    797 				}
    798 			}
    799 		}
    800 		if (IsTakingOff) {
    801 			Altitude++;
    802 			if (Altitude >= FLIGHT_LEVEL) {
    803 				Altitude = FLIGHT_LEVEL;
    804 				IsTakingOff = false;
    805 			}
    806 		}
    807 
    808 		/*
    809 		**	Make adjustments for altitude by moving from one layer to another as
    810 		**	necessary.
    811 		*/
    812 		if (layer != In_Which_Layer()) {
    813 
    814 			/*
    815 			**	When the aircraft is about to enter the ground layer, perform on last
    816 			**	check to see if it is legal to enter that location. If not, then
    817 			**	start the take off process. Let the normal logic handle this
    818 			**	change of plans.
    819 			*/
    820 			bool ok = true;
    821 			if (In_Which_Layer() == LAYER_GROUND) {
    822 				if (!Is_LZ_Clear(::As_Target(Coord_Cell(Coord)))) {
    823 					IsTakingOff = true;
    824 					Altitude++;
    825 					ok = false;
    826 				}
    827 			}
    828 
    829 			if (ok) {
    830 				/*
    831 				**	If landing in a cell that already contains an object, then
    832 				**	the landing attempt must be aborted.
    833 				*/
    834 				Map.Remove(this, layer);
    835 				Map.Submit(this, In_Which_Layer());
    836 
    837 				/*
    838 				**	When the aircraft is close to the ground, it should exist as a ground object.
    839 				**	This aspect is controlled by the Place_Down and Pick_Up functions.
    840 				*/
    841 				if (In_Which_Layer() == LAYER_GROUND) {
    842 					Assign_Destination(TARGET_NONE);		// Clear the navcom.
    843 					Transmit_Message(RADIO_TETHER);
    844 					Map.Place_Down(Coord_Cell(Coord), this);
    845 					//if (IsOwnedByPlayer) {				// Changed for multiple player mapping. ST - 3/6/2019 1:31PM
    846 					if (House->IsHuman) {
    847 						Map.Sight_From(House, Coord_Cell(Coord), 1, false);
    848 					}
    849 				} else  {
    850 					Transmit_Message(RADIO_UNTETHER);
    851 					Map.Pick_Up(Coord_Cell(Coord), this);
    852 
    853 					/*
    854 					**	If the navigation computer is not attached to the object this
    855 					**	aircraft is in radio contact with, then assume that radio
    856 					**	contact is now superfluous. Break radio contact.
    857 					*/
    858 					if (In_Radio_Contact() && Target_Legal(NavCom) && NavCom != Contact_With_Whom()->As_Target()) {
    859 						Transmit_Message(RADIO_OVER_OUT);
    860 					}
    861 				}
    862 			}
    863 		}
    864 	}
    865 
    866 	/*
    867 	**	Always flag the map draw process to occur if there is an aircraft in the view.
    868 	**	This ensures that it will be rendered even if there is nothing else that flagged
    869 	**	the map to be redrawn.
    870 	*/
    871 	if (Map.In_View(Coord_Cell(Coord))) {
    872 		Map.Flag_To_Redraw(false);
    873 		Map.DisplayClass::IsToRedraw = true;
    874 	}
    875 
    876 	/*
    877 	**	When aircraft leave the edge of the map, they might get destroyed. This occurs if the
    878 	**	aircraft is a non-player produced unit and it has completed its mission. A transport
    879 	**	helicopter that has already delivered reinforcements is a good example of this.
    880 	*/
    881 	if (!Map.In_Radar(Coord_Cell(Coord))) {
    882 		if (Mission == MISSION_RETREAT /*|| (*this == AIRCRAFT_CARGO && !Is_Something_Attached())*/) {
    883 
    884 			/*
    885 			**	Check to see if there are any civilians aboard. If so, then flag the house
    886 			**	that the civilian evacuation trigger event has been fulfilled.
    887 			*/
    888 			while (Is_Something_Attached()) {
    889 				FootClass * obj = Detach_Object();
    890 				if (obj->What_Am_I() == RTTI_INFANTRY && ((InfantryClass *)obj)->Class->IsCivilian && !((InfantryClass *)obj)->IsTechnician) {
    891 					House->IsCivEvacuated = true;
    892 				}
    893 
    894 				/*
    895 				**	Transport planes that leave can only be because they carry purchased
    896 				**	equipment and must be have their cost refunded.
    897 				*/
    898 				if (*this == AIRCRAFT_CARGO) {
    899 					House->Refund_Money(obj->Class_Of().Cost_Of());
    900 				}
    901 				delete obj;
    902 			}
    903 			Stun();
    904 			Delete_This();
    905 			return;
    906 		}
    907 	} else {
    908 		IsLocked = true;
    909 //		House->NewAScan |= (1L << Class->Type);
    910 
    911 #ifdef NEVER
    912 		/*
    913 		**	Transport helicopters must ensure that their passengers are properly
    914 		**	considered "alive" by setting the appropriate scan bits.
    915 		*/
    916 		FootClass const * foot = Attached_Object();
    917 		while (foot) {
    918 			switch (foot->What_Am_I()) {
    919 				case RTTI_UNIT:
    920 					House->NewUScan |= (1L << ((UnitTypeClass const &)Class_Of()).Type);
    921 					break;
    922 
    923 				case RTTI_INFANTRY:
    924 					House->NewIScan |= (1L << ((InfantryTypeClass const &)Class_Of()).Type);
    925 					break;
    926 			}
    927 
    928 			foot = (FootClass const *)foot->Next;
    929 		}
    930 #endif
    931 	}
    932 }
    933 
    934 
    935 /***********************************************************************************************
    936  * AircraftClass::Mark -- Flags cells under the aircraft so that they will be redrawn.         *
    937  *                                                                                             *
    938  *    This routine is used to flag the cells under the aircraft so that those cells will       *
    939  *    be redrawn during the next map drawing process. This is a necessary step whenever the    *
    940  *    aircraft moves or changes shape.                                                         *
    941  *                                                                                             *
    942  * INPUT:   none                                                                               *
    943  *                                                                                             *
    944  * OUTPUT:  none                                                                               *
    945  *                                                                                             *
    946  * WARNINGS:   none                                                                            *
    947  *                                                                                             *
    948  * HISTORY:                                                                                    *
    949  *   07/26/1994 JLB : Created.                                                                 *
    950  *=============================================================================================*/
    951 bool AircraftClass::Mark(MarkType mark)
    952 {
    953 	Validate();
    954 	if (FootClass::Mark(mark)) {
    955 		Map.Refresh_Cells(Coord_Cell(Coord), Occupy_List());
    956 		Map.Refresh_Cells(Coord_Cell(Coord), Overlap_List());
    957 		return(true);
    958 	}
    959 	return(false);
    960 }
    961 
    962 
    963 /***********************************************************************************************
    964  * AircraftClass::Overlap_List -- Returns with list of cells the aircraft overlaps.            *
    965  *                                                                                             *
    966  *    When aircraft are flying, they can overlap quite a number of cells. These cells can      *
    967  *    be determined from the coordinate where the aircraft is centered and the size of the     *
    968  *    aircraft's shape. Landed aircraft are a special case and are usually much smaller        *
    969  *    than when flying.                                                                        *
    970  *                                                                                             *
    971  * INPUT:   none                                                                               *
    972  *                                                                                             *
    973  * OUTPUT:  Returns with a pointer to a cell offset list that specifies all cells that         *
    974  *          the aircraft overlaps given the aircraft's current state.                          *
    975  *                                                                                             *
    976  * WARNINGS:   none                                                                            *
    977  *                                                                                             *
    978  * HISTORY:                                                                                    *
    979  *   07/26/1994 JLB : Created.                                                                 *
    980  *=============================================================================================*/
    981 short const *AircraftClass::Overlap_List(void) const
    982 {
    983 	Validate();
    984 	static short const _list[] = {
    985 		-(MAP_CELL_W-1), -MAP_CELL_W, -(MAP_CELL_W+1),
    986 		-1, 0, 1,
    987 		(MAP_CELL_W-1), MAP_CELL_W, (MAP_CELL_W+1),
    988 		-((MAP_CELL_W*2)-1), -(MAP_CELL_W*2), -((MAP_CELL_W*2)+1),
    989 		-((MAP_CELL_W*3)-1), -(MAP_CELL_W*3), -((MAP_CELL_W*3)+1),
    990 		REFRESH_EOL
    991 	};
    992 
    993 	if (Altitude) {
    994 		return(_list);
    995 		//return Coord_Spillage_List(Coord, 25);
    996 	}
    997 	return(Class->Overlap_List());
    998 }
    999 
   1000 
   1001 /***********************************************************************************************
   1002  * AircraftClass::Init -- Initialize the aircraft system to an empty state.                    *
   1003  *                                                                                             *
   1004  *    This routine is used to clear out the aircraft allocation system. It is called in        *
   1005  *    preparation for a scenario load or save game load.                                       *
   1006  *                                                                                             *
   1007  * INPUT:   none                                                                               *
   1008  *                                                                                             *
   1009  * OUTPUT:  none                                                                               *
   1010  *                                                                                             *
   1011  * WARNINGS:   none                                                                            *
   1012  *                                                                                             *
   1013  * HISTORY:                                                                                    *
   1014  *   09/24/1994 JLB : Created.                                                                 *
   1015  *=============================================================================================*/
   1016 void AircraftClass::Init(void)
   1017 {
   1018 	AircraftClass *ptr;
   1019 
   1020 	Aircraft.Free_All();
   1021 
   1022 	ptr = new AircraftClass();
   1023 	VTable = ((void **)(((char *)ptr) + sizeof(AbstractClass) - 4))[0];
   1024 	delete ptr;
   1025 }
   1026 
   1027 
   1028 /***********************************************************************************************
   1029  * AircraftClass::Mission_Unload -- Handles unloading cargo.                                   *
   1030  *                                                                                             *
   1031  *    This function is used to handle finding, heading toward, landing, and unloading the      *
   1032  *    cargo from the aircraft. Once unloading of cargo has occurred, then the aircraft follows *
   1033  *    a different mission.                                                                     *
   1034  *                                                                                             *
   1035  * INPUT:   none                                                                               *
   1036  *                                                                                             *
   1037  * OUTPUT:  Returns the number of game ticks to delay before calling this function again.      *
   1038  *                                                                                             *
   1039  * WARNINGS:   none                                                                            *
   1040  *                                                                                             *
   1041  * HISTORY:                                                                                    *
   1042  *   10/31/94   JLB : Created.                                                                 *
   1043  *=============================================================================================*/
   1044 int AircraftClass::Mission_Unload(void)
   1045 {
   1046 	Validate();
   1047 	if (Class->IsFixedWing) {
   1048 		enum {
   1049 			PICK_AIRSTRIP,
   1050 			FLY_TO_AIRSTRIP,
   1051 			BUG_OUT
   1052 		};
   1053 
   1054 		switch (Status) {
   1055 
   1056 			/*
   1057 			**	Find a suitable airfield to land at.
   1058 			*/
   1059 			case PICK_AIRSTRIP:
   1060 				if (!Target_Legal(NavCom) || !In_Radio_Contact()) {
   1061 
   1062 					BuildingClass * target_building = As_Building(NavCom);
   1063 					BuildingClass * building = (target_building != NULL && *target_building == STRUCT_AIRSTRIP) ? target_building : Find_Docking_Bay(STRUCT_AIRSTRIP, false);
   1064 					if (building) {
   1065 						if (Transmit_Message(RADIO_HELLO, building) == RADIO_ROGER) {
   1066 							Set_Speed(0xFF);
   1067 							Assign_Destination(building->As_Target());
   1068 							if (Team) {
   1069 								Team->Target = NavCom;
   1070 							}
   1071 							Status = FLY_TO_AIRSTRIP;
   1072 						}
   1073 					}
   1074 
   1075 					/*
   1076 					**	If a suitable airfield could not be found, then just randomly change
   1077 					**	direction and then try again later.
   1078 					*/
   1079 					if (Status == PICK_AIRSTRIP) {
   1080 
   1081 						/*
   1082 						**	If there are no more airstrips, regardless of busy state, then
   1083 						**	abort this transport plane completely.
   1084 						*/
   1085 						if (!(House->ActiveBScan & STRUCTF_AIRSTRIP)) {
   1086 							Assign_Mission(MISSION_RETREAT);
   1087 						}
   1088 
   1089 						/*
   1090 						**	Pick a new direction and fly off.
   1091 						*/
   1092 						PrimaryFacing.Set_Desired(Random_Pick(DIR_N, DIR_MAX));
   1093 						SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   1094 						return(TICKS_PER_SECOND*3);
   1095 					}
   1096 				} else {
   1097 					Status = FLY_TO_AIRSTRIP;
   1098 				}
   1099 				break;
   1100 
   1101 			/*
   1102 			**	Home in on target. When close enough, drop the cargo.
   1103 			*/
   1104 			case FLY_TO_AIRSTRIP:
   1105 				if (!Target_Legal(NavCom) || !In_Radio_Contact()) {
   1106 					Status = PICK_AIRSTRIP;
   1107 				} else {
   1108 
   1109 					/*
   1110 					**	If, for some reason, there is no cargo, then don't stick around.
   1111 					*/
   1112 					if (!Is_Something_Attached()) {
   1113 						Status = BUG_OUT;
   1114 						return(1);
   1115 					}
   1116 
   1117 					if (!PrimaryFacing.Is_Rotating()) {
   1118 						PrimaryFacing.Set_Desired(Direction(As_Movement_Coord(NavCom)));
   1119 					}
   1120 					SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   1121 
   1122 					int navdist = Distance(As_Movement_Coord(NavCom));
   1123 					Altitude = FLIGHT_LEVEL;
   1124 					if (navdist < 0x0600) {
   1125 						Altitude = Fixed_To_Cardinal(FLIGHT_LEVEL, Cardinal_To_Fixed(0x0600, navdist));
   1126 					}
   1127 
   1128 					if (navdist < 0x0080) {
   1129 						FootClass * unit = (FootClass *)Detach_Object();
   1130 
   1131 						if (unit) {
   1132 							CELL cell = Contact_With_Whom()->Find_Exit_Cell(unit);
   1133 							if (cell) {
   1134 								ScenarioInit++;
   1135 								if (!unit->Unlimbo(Cell_Coord(cell))) {
   1136 									Attach(unit);
   1137 								} else {
   1138 
   1139 									/*
   1140 									**	Cargo planes announce reinforcements when they unload
   1141 									**	their cargo.
   1142 									*/
   1143 									if (*this == AIRCRAFT_CARGO && House == PlayerPtr) {
   1144 										Speak(VOX_REINFORCEMENTS, NULL, Cell_Coord(cell));
   1145 									}
   1146 									unit->IsALoaner = false;
   1147 									unit->IsLocked = true;
   1148 									unit->Scatter(0, true);
   1149 								}
   1150 								ScenarioInit--;
   1151 
   1152 								Transmit_Message(RADIO_OVER_OUT);
   1153 								Assign_Target(TARGET_NONE);
   1154 							} else {
   1155 								Attach(unit);
   1156 							}
   1157 
   1158 //							if (Is_Something_Attached()) {
   1159 //								Status = PICK_AIRSTRIP;
   1160 //							} else {
   1161 								Status = BUG_OUT;
   1162 //							}
   1163 						} else {
   1164 							Status = BUG_OUT;
   1165 						}
   1166 					}
   1167 					return(1);
   1168 				}
   1169 				break;
   1170 
   1171 			/*
   1172 			**	All cargo unloaded, head off the map.
   1173 			*/
   1174 			case BUG_OUT:
   1175 				Assign_Mission(MISSION_RETREAT);
   1176 				return(1);
   1177 		}
   1178 
   1179 	} else {
   1180 		enum {
   1181 			SEARCH_FOR_LZ,
   1182 			FLY_TO_LZ,
   1183 			LAND_ON_LZ,
   1184 			UNLOAD_PASSENGERS,
   1185 			TAKE_OFF,
   1186 		};
   1187 
   1188 		switch (Status) {
   1189 
   1190 			/*
   1191 			**	Search for an appropriate destination spot if one isn't already assigned.
   1192 			*/
   1193 			case SEARCH_FOR_LZ:
   1194 				if (Altitude == 0 && (Target_Legal(NavCom) || Coord == As_Coord(NavCom))) {
   1195 					Status = UNLOAD_PASSENGERS;
   1196 				} else {
   1197 					if (!Is_LZ_Clear(NavCom)) {
   1198 						Assign_Destination(New_LZ(::As_Target(Waypoint[WAYPT_REINF])));
   1199 					} else {
   1200 						if (Altitude == FLIGHT_LEVEL) {
   1201 							Status = FLY_TO_LZ;
   1202 						} else {
   1203 							Status = TAKE_OFF;
   1204 						}
   1205 					}
   1206 				}
   1207 				break;
   1208 
   1209 			/*
   1210 			**	Fly to destination.
   1211 			*/
   1212 			case FLY_TO_LZ:
   1213 				if (Is_LZ_Clear(NavCom)) {
   1214 					int distance = Process_Fly_To(true);
   1215 
   1216 					if (distance < 0x0100) {
   1217 						SecondaryFacing.Set_Desired(Pose_Dir());
   1218 
   1219 						if (distance < 0x0010) {
   1220 							Status = LAND_ON_LZ;
   1221 						}
   1222 						return(1);
   1223 					} else {
   1224 						SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   1225 						return(5);
   1226 					}
   1227 				} else {
   1228 					Status = SEARCH_FOR_LZ;
   1229 				}
   1230 				break;
   1231 
   1232 			/*
   1233 			**	Landing phase. Just delay until landing is complete. At that time,
   1234 			**	transition to the unloading phase.
   1235 			*/
   1236 			case LAND_ON_LZ:
   1237 				if (IsTakingOff) {
   1238 					Status = TAKE_OFF;
   1239 				} else {
   1240 					if (Process_Landing()) {
   1241 						Status = UNLOAD_PASSENGERS;
   1242 					}
   1243 				}
   1244 				return(1);
   1245 
   1246 			/*
   1247 			**	Hold while unloading passengers. When passengers are unloaded the order for this
   1248 			**	transport gets changed to MISSION_RETREAT.
   1249 			*/
   1250 			case UNLOAD_PASSENGERS:
   1251 				if (!IsTethered) {
   1252 					if (Is_Something_Attached()) {
   1253 						FootClass * unit = (FootClass *)Detach_Object();
   1254 
   1255 						/*
   1256 						**	First thing is to lift the transport off of the map so that the unlimbo
   1257 						**	process for the passengers is more likely to succeed.
   1258 						*/
   1259 						Map.Pick_Up(Coord_Cell(Coord), this);
   1260 
   1261 						if (!Exit_Object(unit)) {
   1262 							delete unit;
   1263 						}
   1264 
   1265 						/*
   1266 						**	Restore the transport back down on the map.
   1267 						*/
   1268 						Map.Place_Down(Coord_Cell(Coord), this);
   1269 
   1270 						if (!Is_Something_Attached()) {
   1271 							Enter_Idle_Mode();
   1272 						}
   1273 
   1274 					} else  {
   1275 
   1276 						Enter_Idle_Mode();
   1277 					}
   1278 				}
   1279 				break;
   1280 
   1281 			/*
   1282 			**	Aircraft is now taking off. Once the aircraft reaches flying altitude then it
   1283 			**	will either take off or look for another landing spot to try again.
   1284 			*/
   1285 			case TAKE_OFF:  {
   1286 				if (Process_Take_Off()) {
   1287 					if (Is_Something_Attached()) {
   1288 						Status = SEARCH_FOR_LZ;
   1289 
   1290 						/*
   1291 						**	Break off radio contact with the helipad it is taking off from.
   1292 						*/
   1293 						if (In_Radio_Contact() && Map[Coord_Cell(Coord)].Cell_Building() == Contact_With_Whom()) {
   1294 							Transmit_Message(RADIO_OVER_OUT);
   1295 						}
   1296 					} else  {
   1297 						Enter_Idle_Mode();
   1298 					}
   1299 				}
   1300 				return(1);
   1301 			}
   1302 		}
   1303 	}
   1304 	return(10);
   1305 }
   1306 
   1307 
   1308 /***********************************************************************************************
   1309  * AircraftClass::Is_LZ_Clear -- Determines if landing zone is free for landing.               *
   1310  *                                                                                             *
   1311  *    This routine examines the landing zone (as specified by the target parameter) in order   *
   1312  *    to determine if it is free to be landed upon. Call this routine when it is necessary     *
   1313  *    to double check this. Typically this occurs right before a helicopter lands and also     *
   1314  *    when determining the landing zone in the first place.                                    *
   1315  *                                                                                             *
   1316  * INPUT:   target   -- The target that is the "landing zone".                                 *
   1317  *                                                                                             *
   1318  * OUTPUT:  bool; Is the landing zone clear for landing?                                       *
   1319  *                                                                                             *
   1320  * WARNINGS:   none                                                                            *
   1321  *                                                                                             *
   1322  * HISTORY:                                                                                    *
   1323  *   10/31/94   JLB : Created.                                                                 *
   1324  *=============================================================================================*/
   1325 bool AircraftClass::Is_LZ_Clear(TARGET target) const
   1326 {
   1327 	Validate();
   1328 	if (!Target_Legal(target)) return(false);
   1329 	CELL cell = ::As_Cell(target);
   1330 	if (!Map.In_Radar(cell)) return(false);
   1331 
   1332 	ObjectClass * object = Map[cell].Cell_Object();
   1333 	if (object) {
   1334 		if (object == this) return(true);
   1335 
   1336 		if (In_Radio_Contact() && Contact_With_Whom() == object) {
   1337 			return(true);
   1338 		}
   1339 		return(false);
   1340 	}
   1341 
   1342 	if (!Map[cell].Is_Generally_Clear()) return(false);
   1343 
   1344 	return(true);
   1345 }
   1346 
   1347 
   1348 /***********************************************************************************************
   1349  * AircraftClass::In_Which_Layer -- Determine which render layer the aircraft lies.            *
   1350  *                                                                                             *
   1351  *    This routine is used to figure out which rendering layer the aircraft is located in.     *
   1352  *    It can be determined from the aircraft's height. The layer value is used to handle the   *
   1353  *    display sequence. Objects in lower layers appear beneath objects in higher layers.       *
   1354  *                                                                                             *
   1355  * INPUT:   none                                                                               *
   1356  *                                                                                             *
   1357  * OUTPUT:  Returns the layer that the aircraft is currently located in.                       *
   1358  *                                                                                             *
   1359  * WARNINGS:   none                                                                            *
   1360  *                                                                                             *
   1361  * HISTORY:                                                                                    *
   1362  *   11/01/1994 JLB : Created.                                                                 *
   1363  *=============================================================================================*/
   1364 LayerType AircraftClass::In_Which_Layer(void) const
   1365 {
   1366 	Validate();
   1367 	if (Class->IsFixedWing) return(LAYER_TOP);
   1368 
   1369 	if (Altitude < FLIGHT_LEVEL - (FLIGHT_LEVEL/3)) {
   1370 		return(LAYER_GROUND);
   1371 	}
   1372 	return(LAYER_TOP);
   1373 }
   1374 
   1375 
   1376 /***********************************************************************************************
   1377  * AircraftClass::Sort_Y -- Figures the sorting coordinate.                                    *
   1378  *                                                                                             *
   1379  *    This routine is used to determine the coordinate to use for sorting the aircraft. This   *
   1380  *    sorting value is used when the aircraft is on the ground. At that time the aircraft      *
   1381  *    must be rendered in proper relationship to the other ground objects.                     *
   1382  *                                                                                             *
   1383  * INPUT:   none                                                                               *
   1384  *                                                                                             *
   1385  * OUTPUT:  Returns with the coordinate to use when sorting the aircraft with other ground     *
   1386  *          objects.                                                                           *
   1387  *                                                                                             *
   1388  * WARNINGS:   none                                                                            *
   1389  *                                                                                             *
   1390  * HISTORY:                                                                                    *
   1391  *   11/02/1994 JLB : Created.                                                                 *
   1392  *=============================================================================================*/
   1393 COORDINATE AircraftClass::Sort_Y(void) const
   1394 {
   1395 	Validate();
   1396 	return(Coord_Add(Coord, 0x00800000L));
   1397 }
   1398 
   1399 
   1400 /***********************************************************************************************
   1401  * AircraftClass::Mission_Retreat -- Handles the aircraft logic for leaving the battlefield.   *
   1402  *                                                                                             *
   1403  *    This mission will be followed when the aircraft decides that it is time to leave the     *
   1404  *    battle. Typically, this occurs when a loaner transport has dropped off its load or when  *
   1405  *    an attack air vehicle has expended its ordinance.                                        *
   1406  *                                                                                             *
   1407  * INPUT:   none                                                                               *
   1408  *                                                                                             *
   1409  * OUTPUT:  Returns with the number of game ticks to delay before calling this routine again.  *
   1410  *                                                                                             *
   1411  * WARNINGS:   none                                                                            *
   1412  *                                                                                             *
   1413  * HISTORY:                                                                                    *
   1414  *   03/19/1995 JLB : Created.                                                                 *
   1415  *   08/13/1995 JLB : Handles aircraft altitude gain after takeoff logic.                      *
   1416  *=============================================================================================*/
   1417 int AircraftClass::Mission_Retreat(void)
   1418 {
   1419 	Validate();
   1420 	if (Class->IsFixedWing) {
   1421 		if (*this == AIRCRAFT_CARGO) {
   1422 			PrimaryFacing.Set_Desired(DIR_W);
   1423 			SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   1424 		}
   1425 		if (Altitude < FLIGHT_LEVEL) {
   1426 			Altitude++;
   1427 			return(1);
   1428 		}
   1429 		return(TICKS_PER_SECOND*10);
   1430 	}
   1431 
   1432 	enum {
   1433 		TAKE_OFF,
   1434 		FACE_MAP_EDGE,
   1435 		KEEP_FLYING
   1436 	};
   1437 	switch (Status) {
   1438 
   1439 		/*
   1440 		**	Take off if landed.
   1441 		*/
   1442 		case TAKE_OFF:
   1443 			if (Process_Take_Off()) {
   1444 				Status = FACE_MAP_EDGE;
   1445 			}
   1446 			return(1);
   1447 
   1448 		/*
   1449 		**	Set facing and speed toward the friendly map edge.
   1450 		*/
   1451 		case FACE_MAP_EDGE:
   1452 			Set_Speed(0xFF);
   1453 
   1454 			/*
   1455 			**	Take advantage of the fact that the source map edge enumerations happen to
   1456 			**	occur in a clockwise order and are the first four enumerations of the map
   1457 			**	edge default for the house. If this value is masked and then shifted, a
   1458 			**	normalized direction value results. Use this value to head the aircraft
   1459 			**	toward the "friendly" map edge.
   1460 			*/
   1461 			PrimaryFacing.Set_Desired((DirType)((House->Edge & 0x03) << 6));
   1462 			SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   1463 			Status = KEEP_FLYING;
   1464 			break;
   1465 
   1466 		/*
   1467 		**	Just do nothing since we are headed toward the map edge. When the edge is
   1468 		**	reached, the aircraft should be automatically eliminated.
   1469 		*/
   1470 		case KEEP_FLYING:
   1471 			break;
   1472 
   1473 		default:
   1474 			break;
   1475 	}
   1476 	return(TICKS_PER_SECOND);
   1477 }
   1478 
   1479 
   1480 /***********************************************************************************************
   1481  * AircraftClass::Exit_Object -- Unloads passenger from aircraft.                              *
   1482  *                                                                                             *
   1483  *    This routine is called when the aircraft is to unload a passenger. The passenger must    *
   1484  *    be able to move under its own power. Typical situation is when a transport helicopter    *
   1485  *    is to unload an infantry unit.                                                           *
   1486  *                                                                                             *
   1487  * INPUT:   unit  -- Pointer to the unit that is to be unloaded from this aircraft.            *
   1488  *                                                                                             *
   1489  * OUTPUT:  bool; Was the unit unloaded successfully?                                          *
   1490  *                                                                                             *
   1491  * WARNINGS:   The unload process is merely started by this routine. Radio contact is          *
   1492  *             established with the unloading unit and when the unit is clear of the aircraft  *
   1493  *             the radio contact will be broken and then the aircraft is free to pursue        *
   1494  *             other.                                                                          *
   1495  *                                                                                             *
   1496  * HISTORY:                                                                                    *
   1497  *   01/10/1995 JLB : Created.                                                                 *
   1498  *=============================================================================================*/
   1499 int AircraftClass::Exit_Object(TechnoClass * unit)
   1500 {
   1501 	Validate();
   1502 	static FacingType _toface[FACING_COUNT] = {FACING_S, FACING_SW, FACING_SE, FACING_NW, FACING_NE, FACING_N, FACING_W, FACING_E};
   1503 	CELL	cell;
   1504 
   1505 	/*
   1506 	**	Find a free cell to drop the unit off at.
   1507 	*/
   1508 	FacingType face;
   1509 	for (face = FACING_N; face < FACING_COUNT; face++) {
   1510 		cell = Adjacent_Cell(Coord_Cell(Coord), _toface[face]);
   1511 		if (unit->Can_Enter_Cell(cell) == MOVE_OK) break;
   1512 	}
   1513 
   1514 	// Should perform a check here to see if no cell could be found.
   1515 
   1516 	/*
   1517 	**	If the passenger can be placed on the map, then start it moving toward the
   1518 	**	destination cell and establish radio contact with the transport. This is used
   1519 	**	to make sure that the transport waits until the passenger is clear before
   1520 	**	unloading the next passenger or taking off.
   1521 	*/
   1522 	if (unit->Unlimbo(Coord, Facing_Dir(_toface[face]))) {
   1523 		unit->Assign_Mission(MISSION_MOVE);
   1524 		unit->Assign_Destination(::As_Target(cell));
   1525 		if (Transmit_Message(RADIO_HELLO, unit) == RADIO_ROGER) {
   1526 			Transmit_Message(RADIO_UNLOAD);
   1527 		}
   1528 		unit->Look(false);
   1529 		return(1);
   1530 	}
   1531 	return(0);
   1532 }
   1533 
   1534 
   1535 /***********************************************************************************************
   1536  * AircraftClass::Fire_At -- Handles firing a projectile from an aircraft.                     *
   1537  *                                                                                             *
   1538  *    Sometimes, aircraft firing needs special handling. Example: for napalm bombs, the        *
   1539  *    bomb travels forward at nearly the speed of the delivery aircraft, not necessarily the   *
   1540  *    default speed defined in the BulletTypeClass structure.                                  *
   1541  *                                                                                             *
   1542  * INPUT:   target   -- The target that the projectile is heading for.                         *
   1543  *                                                                                             *
   1544  *          which    -- Which weapon to use in the attack. 0=primary, 1=secondary.             *
   1545  *                                                                                             *
   1546  * OUTPUT:  Returns with a pointer to the bullet that was created as a result of this attack.  *
   1547  *                                                                                             *
   1548  * WARNINGS:   none                                                                            *
   1549  *                                                                                             *
   1550  * HISTORY:                                                                                    *
   1551  *   03/19/1995 JLB : Created.                                                                 *
   1552  *=============================================================================================*/
   1553 BulletClass * AircraftClass::Fire_At(TARGET target, int which)
   1554 {
   1555 	Validate();
   1556 	BulletClass * bullet = FootClass::Fire_At(target, which);
   1557 
   1558 	if (bullet) {
   1559 
   1560 		/*
   1561 		**	Aircraft reveal when firing
   1562 		*/
   1563 		HouseClass *player = HouseClass::As_Pointer(Owner());
   1564 		if (player != nullptr && player->IsHuman) {
   1565 			Map.Sight_From(player, Coord_Cell(Center_Coord()), 1, false);
   1566 		}
   1567 
   1568 		/*
   1569 		**	Play the sound effect associated with this weapon.
   1570 		*/
   1571 		WeaponTypeClass const * weapon = (which == 0) ? &Weapons[Class->Primary] : &Weapons[Class->Secondary];
   1572 		Sound_Effect(weapon->Sound, Coord);
   1573 
   1574 
   1575 		/*
   1576 		**	Falling bullets move at a speed proportionate to the delivery craft.
   1577 		*/
   1578 		if (bullet->Class->IsDropping) {
   1579 			bullet->Fly_Speed(40, bullet->Class->MaxSpeed);
   1580 		}
   1581 	}
   1582 	return(bullet);
   1583 }
   1584 
   1585 
   1586 /***********************************************************************************************
   1587  * AircraftClass::Take_Damage -- Applies damage to the aircraft.                               *
   1588  *                                                                                             *
   1589  *    This routine is used to apply damage to the specified aircraft. This is where any        *
   1590  *    special crash animation will be initiated.                                               *
   1591  *                                                                                             *
   1592  * INPUT:   damage   -- Reference to the damage that will be applied to the aircraft.          *
   1593  *                      This value will be filled in with the actual damage that was           *
   1594  *                      applied.                                                               *
   1595  *                                                                                             *
   1596  *          distance -- Distance from the source of the explosion to this aircraft.            *
   1597  *                                                                                             *
   1598  *          warhead  -- The warhead type that the damage occurs from.                          *
   1599  *                                                                                             *
   1600  *          source   -- Pointer to the originator of the damage. This can be used so that      *
   1601  *                      proper "thank you" can be delivered.                                   *
   1602  *                                                                                             *
   1603  * OUTPUT:  Returns with the result of the damage as it affects this aircraft.                 *
   1604  *                                                                                             *
   1605  * WARNINGS:   none                                                                            *
   1606  *                                                                                             *
   1607  * HISTORY:                                                                                    *
   1608  *   05/26/1995 JLB : Created.                                                                 *
   1609  *=============================================================================================*/
   1610 ResultType AircraftClass::Take_Damage(int & damage, int distance, WarheadType warhead, TechnoClass * source)
   1611 {
   1612 	Validate();
   1613 	ResultType res = RESULT_NONE;
   1614 
   1615 	/*
   1616 	**	Flying aircraft take half damage.
   1617 	*/
   1618 	if (Altitude) {
   1619 		damage /= 2;
   1620 	}
   1621 
   1622 	/*
   1623 	**	In order for a this to be damaged, it must either be a unit
   1624 	**	with a crew or a sandworm.
   1625 	*/
   1626 	res = FootClass::Take_Damage(damage, distance, warhead, source);
   1627 
   1628 	switch (res) {
   1629 		case RESULT_DESTROYED:
   1630 			{
   1631 				Kill_Cargo(source);
   1632 				Death_Announcement();
   1633 				COORDINATE coord = Target_Coord();
   1634 				if (!(coord & 0xC000C000L)) {
   1635 					new AnimClass(ANIM_FBALL1, coord);
   1636 				}
   1637 				Delete_This();
   1638 			}
   1639 			break;
   1640 
   1641 		default:
   1642 		case RESULT_HALF:
   1643 			break;
   1644 	}
   1645 
   1646 	return(res);
   1647 }
   1648 
   1649 
   1650 /***********************************************************************************************
   1651  * AircraftClass::Mission_Move -- Handles movement mission.                                    *
   1652  *                                                                                             *
   1653  *    This state machine routine is used when an aircraft (usually helicopter) is to move      *
   1654  *    from one location to another. It will handle any necessary take off and landing this     *
   1655  *    may require.                                                                             *
   1656  *                                                                                             *
   1657  * INPUT:   none                                                                               *
   1658  *                                                                                             *
   1659  * OUTPUT:  Returns with the number of game frames that should elapse before this routine      *
   1660  *          is called again.                                                                   *
   1661  *                                                                                             *
   1662  * WARNINGS:   none                                                                            *
   1663  *                                                                                             *
   1664  * HISTORY:                                                                                    *
   1665  *   06/19/1995 JLB : Created.                                                                 *
   1666  *=============================================================================================*/
   1667 int AircraftClass::Mission_Move(void)
   1668 {
   1669 	Validate();
   1670 	if (Class->IsFixedWing) {
   1671 
   1672 		/*
   1673 		**	Force aircraft in movement mission into a retreat
   1674 		**	mission so that it leaves the map.
   1675 		*/
   1676 		if (*this == AIRCRAFT_A10) {
   1677 			Assign_Mission(MISSION_RETREAT);
   1678 			Commence();
   1679 			return(1);
   1680 		}
   1681 
   1682 		enum {
   1683 			FLY_TO_AIRSTRIP,
   1684 			BUG_OUT
   1685 		};
   1686 		switch (Status) {
   1687 			/*
   1688 			**	Home in on target. When close enough, drop the cargo.
   1689 			*/
   1690 			case FLY_TO_AIRSTRIP:
   1691 				if (!Target_Legal(NavCom) || !In_Radio_Contact()) {
   1692 					return(TICKS_PER_SECOND);
   1693 				} else {
   1694 
   1695 					/*
   1696 					**	If, for some reason, there is no cargo, then don't stick around.
   1697 					*/
   1698 					if (!Is_Something_Attached()) {
   1699 						Status = BUG_OUT;
   1700 						return(1);
   1701 					}
   1702 
   1703 
   1704 					if (!PrimaryFacing.Is_Rotating()) {
   1705 						PrimaryFacing.Set_Desired(Direction(NavCom));
   1706 					}
   1707 					SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   1708 					if (Distance(NavCom) < 0x0080) {
   1709 						FootClass * unit = (FootClass *)Detach_Object();
   1710 
   1711 						if (unit) {
   1712 							ScenarioInit++;
   1713 							if (!unit->Unlimbo(Coord_Snap(Contact_With_Whom()->Coord))) {
   1714 								Attach(unit);
   1715 							}
   1716 							ScenarioInit--;
   1717 
   1718 							Transmit_Message(RADIO_OVER_OUT);
   1719 							Assign_Target(TARGET_NONE);
   1720 						}
   1721 						Status = BUG_OUT;
   1722 					}
   1723 				}
   1724 				return(1);
   1725 
   1726 			case BUG_OUT:
   1727 				return(TICKS_PER_SECOND);
   1728 		}
   1729 		return(5);
   1730 	}
   1731 
   1732 	enum {
   1733 		VALIDATE_LZ,
   1734 		TAKE_OFF,
   1735 		FLY_TO_LZ,
   1736 		LAND
   1737 	};
   1738 	switch (Status) {
   1739 
   1740 		/*
   1741 		**	Double check and change LZ if necessary.
   1742 		*/
   1743 		case VALIDATE_LZ:
   1744 			if (!Target_Legal(NavCom)) {
   1745 				Enter_Idle_Mode();
   1746 			} else {
   1747 				if (!Is_LZ_Clear(NavCom) || !Cell_Seems_Ok(As_Cell(NavCom))) {
   1748 					Assign_Destination(New_LZ(NavCom));
   1749 				} else {
   1750 					Status = TAKE_OFF;
   1751 				}
   1752 			}
   1753 			break;
   1754 
   1755 		/*
   1756 		**	Take off if necessary.
   1757 		*/
   1758 		case TAKE_OFF:
   1759 			if (!Target_Legal(NavCom)) {
   1760 				Status = VALIDATE_LZ;
   1761 			} else {
   1762 				if (Process_Take_Off()) {
   1763 					/*
   1764 					**	After takeoff is complete, break radio contact with any helipad that this
   1765 					**	helicopter is taking off from.
   1766 					*/
   1767 					if (In_Radio_Contact() && Map[Coord_Cell(Coord)].Cell_Building() == Contact_With_Whom()) {
   1768 						Transmit_Message(RADIO_OVER_OUT);
   1769 					}
   1770 
   1771 					Status = FLY_TO_LZ;
   1772 				}
   1773 				return(1);
   1774 			}
   1775 			break;
   1776 
   1777 		/*
   1778 		**	Fly toward target.
   1779 		*/
   1780 		case FLY_TO_LZ:
   1781 			if (Is_LZ_Clear(NavCom)) {
   1782 				int distance = Process_Fly_To(true);
   1783 
   1784 				if (distance < 0x0080) {
   1785 					if (Target_Legal(TarCom)) {
   1786 						SecondaryFacing.Set_Desired(Direction(TarCom));
   1787 					} else {
   1788 						SecondaryFacing.Set_Desired(Pose_Dir());
   1789 					}
   1790 
   1791 					if (distance < 0x0010) {
   1792 						Status = LAND;
   1793 					}
   1794 					return(1);
   1795 				}
   1796 
   1797 				SecondaryFacing.Set_Desired(::Direction(Fire_Coord(0), As_Coord(NavCom)));
   1798 			} else {
   1799 				Assign_Destination(New_LZ(NavCom));
   1800 				if (!Target_Legal(NavCom)) {
   1801 					Status = LAND;
   1802 				}
   1803 			}
   1804 			return(1);
   1805 
   1806 		/*
   1807 		**	Land on target.
   1808 		*/
   1809 		case LAND:
   1810 			if (IsTakingOff) {
   1811 				Assign_Destination(New_LZ(NavCom));
   1812 				Status = TAKE_OFF;
   1813 			}
   1814 			if (Process_Landing()) {
   1815 				if (MissionQueue == MISSION_NONE) {
   1816 					Enter_Idle_Mode();
   1817 				}
   1818 			}
   1819 			return(1);
   1820 	}
   1821 
   1822 	return(TICKS_PER_SECOND);
   1823 }
   1824 
   1825 
   1826 /***********************************************************************************************
   1827  * AircraftClass::Enter_Idle_Mode -- Gives the aircraft an appropriate mission.                *
   1828  *                                                                                             *
   1829  *    Use this routine when the mission for the aircraft is in doubt. This routine will find   *
   1830  *    an appropriate mission for the aircraft and dispatch it.                                 *
   1831  *                                                                                             *
   1832  * INPUT:   initial  -- Is this called when the unit just leaves a factory or is initially     *
   1833  *                      or is initially placed on the map?                                     *
   1834  *                                                                                             *
   1835  * OUTPUT:  none                                                                               *
   1836  *                                                                                             *
   1837  * WARNINGS:   none                                                                            *
   1838  *                                                                                             *
   1839  * HISTORY:                                                                                    *
   1840  *   06/05/1995 JLB : Created.                                                                 *
   1841  *=============================================================================================*/
   1842 void AircraftClass::Enter_Idle_Mode(bool )
   1843 {
   1844 	Validate();
   1845 	MissionType mission = MISSION_GUARD;
   1846 	if (In_Which_Layer() == LAYER_GROUND) {
   1847 		if (IsALoaner) {
   1848 			if (Is_Something_Attached()) {
   1849 				mission = MISSION_UNLOAD;
   1850 			} else {
   1851 				if (Team) {
   1852 					Team->Remove(this);
   1853 				}
   1854 				mission = MISSION_RETREAT;
   1855 			}
   1856 		} else {
   1857 #ifdef NEVER
   1858 			if (In_Radio_Contact() && Contact_With_Whom() == Map[Coord_Cell(Coord)].Cell_Techno()) {
   1859 				Transmit_Message(RADIO_IM_IN);
   1860 			}
   1861 #endif
   1862 			Assign_Destination(TARGET_NONE);
   1863 			Assign_Target(TARGET_NONE);
   1864 			mission = MISSION_GUARD;
   1865 		}
   1866 	} else {
   1867 		if (Is_Something_Attached()) {
   1868 			if (IsALoaner) {
   1869 				if (Team) {
   1870 					mission = MISSION_GUARD;
   1871 				} else {
   1872 					mission = MISSION_UNLOAD;
   1873 					Assign_Destination(Good_LZ());
   1874 				}
   1875 			} else {
   1876 				Assign_Destination(Good_LZ());
   1877 				mission = MISSION_MOVE;
   1878 			}
   1879 		} else {
   1880 
   1881 			/*
   1882 			**	If this transport is a loaner and part of a team, then remove it from
   1883 			**	the team it is attached to.
   1884 			*/
   1885 			if (IsALoaner) {
   1886 				if (Team) {
   1887 					Team->Remove(this);
   1888 				}
   1889 			}
   1890 
   1891 			if (Class->Primary != WEAPON_NONE) {
   1892 
   1893 				/*
   1894 				**	Weapon equipped helicopters that run out of ammo and were
   1895 				**	brought in as reinforcements will leave the map.
   1896 				*/
   1897 				if (Ammo == 0 && !House->IsHuman && IsALoaner) {
   1898 					mission = MISSION_RETREAT;
   1899 				} else {
   1900 
   1901 					/*
   1902 					**	Continue with the current helipad if there is one.
   1903 					*/
   1904 					if (!In_Radio_Contact() ||
   1905 						Contact_With_Whom()->What_Am_I() != RTTI_BUILDING ||
   1906 						*((BuildingClass*)Contact_With_Whom()) != STRUCT_HELIPAD) {
   1907 						/*
   1908 						**	Normal aircraft try to find a good landing spot to rest.
   1909 						*/
   1910 						BuildingClass * building = Find_Docking_Bay(STRUCT_HELIPAD, false);
   1911 						Assign_Destination(TARGET_NONE);
   1912 						if (building && Transmit_Message(RADIO_HELLO, building) == RADIO_ROGER) {
   1913 							mission = MISSION_ENTER;
   1914 						} else {
   1915 							Assign_Destination(Good_LZ());
   1916 							mission = MISSION_MOVE;
   1917 						}
   1918 					} else {
   1919 						mission = MISSION_ENTER;
   1920 					}
   1921 				}
   1922 			} else {
   1923 				if (Team) return;
   1924 
   1925 				Assign_Destination(Good_LZ());
   1926 				mission = MISSION_MOVE;
   1927 			}
   1928 		}
   1929 	}
   1930 	Assign_Mission(mission);
   1931 	Commence();
   1932 }
   1933 
   1934 
   1935 /***********************************************************************************************
   1936  * AircraftClass::Process_Fly_To -- Handles state machine for flying to destination.           *
   1937  *                                                                                             *
   1938  *    This support routine is used when the helicopter is to fly to the destination. It can    *
   1939  *    optionally slow the helicopter down as it approaches the destination.                    *
   1940  *                                                                                             *
   1941  * INPUT:   slowdown -- Should the aircraft be slowed down when it approaches the dest?        *
   1942  *                                                                                             *
   1943  * OUTPUT:  Returns with the distance remaining between the aircraft and the destination.      *
   1944  *                                                                                             *
   1945  * WARNINGS:   Because the aircraft can be move at a fast speed, the distance to target value  *
   1946  *             will probably never be zero. The likely case will be that the aircraft          *
   1947  *             overshoots the target.                                                          *
   1948  *                                                                                             *
   1949  * HISTORY:                                                                                    *
   1950  *   06/14/1995 JLB : Created.                                                                 *
   1951  *=============================================================================================*/
   1952 int AircraftClass::Process_Fly_To(bool slowdown)
   1953 {
   1954 	Validate();
   1955 	COORDINATE coord;
   1956 	if (Is_Target_Building(NavCom)) {
   1957 		coord = As_Building(NavCom)->Docking_Coord();
   1958 	} else {
   1959 		coord = As_Coord(NavCom);
   1960 	}
   1961 	int distance = Distance(coord);
   1962 
   1963 	PrimaryFacing.Set_Desired(Direction(coord));
   1964 
   1965 	if (slowdown) {
   1966 		int speed = MIN(distance, 0x0300);
   1967 		speed = Bound(speed/3, 0x0020, 0x00FF);
   1968 		if (Speed != speed) {
   1969 			Set_Speed(speed);
   1970 		}
   1971 	}
   1972 
   1973 	if (distance < 0x0010) {
   1974 		if (slowdown) {
   1975 			Set_Speed(0);
   1976 		}
   1977 		distance = 0;
   1978 	}
   1979 	return(distance);
   1980 }
   1981 
   1982 
   1983 #ifdef CHEAT_KEYS
   1984 /***********************************************************************************************
   1985  * AircraftClass::Debug_Dump -- Displays the status of the aircraft to the mono monitor.       *
   1986  *                                                                                             *
   1987  *    This displays the current status of the aircraft class to the mono monitor. By this      *
   1988  *    display bugs may be tracked down or prevented.                                           *
   1989  *                                                                                             *
   1990  * INPUT:   none                                                                               *
   1991  *                                                                                             *
   1992  * OUTPUT:  none                                                                               *
   1993  *                                                                                             *
   1994  * WARNINGS:   none                                                                            *
   1995  *                                                                                             *
   1996  * HISTORY:                                                                                    *
   1997  *   06/02/1994 JLB : Created.                                                                 *
   1998  *=============================================================================================*/
   1999 void AircraftClass::Debug_Dump(MonoClass *mono) const
   2000 {
   2001 	Validate();
   2002 	mono->Set_Cursor(0, 0);
   2003 	mono->Print(
   2004 		"�Name:���������������Mission:����TarCom:�NavCom:�Radio:�Coord:���Altitude�St:Ŀ\n"
   2005 		"�                   �           �       �       �      �        �        �    �\n"
   2006 		"����������������N�Y�Health:��Fdir:��Bdir:��Speed:��������������Cargo:��������Ĵ\n"
   2007 		"�Active........� � �        �     �       �      �            �               �\n"
   2008 		"�Limbo.........� � ����������������������������������������������������������Ĵ\n"
   2009 		"�Owned.........� � �Last Message:                                             �\n"
   2010 		"�Discovered....� � �Timer:�Arm:������������������Flash:�Stage:�Team:�����Arch:�\n"
   2011 		"�Selected......� � �      �    �      �         �      �      �         �     �\n"
   2012 		"�Teathered.....� � ������������������������������������������������������������\n"
   2013 		"�Locked on Map.� � �                                                           \n"
   2014 		"�              � � �                                                           \n"
   2015 		"�Is A Loaner...� � �                                                           \n"
   2016 		"�Is Landing....� � �                                                           \n"
   2017 		"�Is Taking Off.� � �                                                           \n"
   2018 		"�              � � �                                                           \n"
   2019 		"�              � � �                                                           \n"
   2020 		"�              � � �                                                           \n"
   2021 		"�Recoiling.....� � �                                                           \n"
   2022 		"�To Display....� � �                                                           \n"
   2023 		"��������������������                                                           \n");
   2024 	mono->Set_Cursor(1, 1);mono->Printf("%s:%s", House->Class->IniName, Class->IniName);
   2025 	mono->Set_Cursor(36, 3);mono->Printf("%02X:%02X", SecondaryFacing.Current(), SecondaryFacing.Desired());
   2026 	mono->Set_Cursor(42, 1);mono->Printf("%04X", NavCom);
   2027 	mono->Set_Cursor(66, 1);mono->Printf("%d", Altitude);
   2028 	mono->Set_Cursor(44, 3);mono->Printf("%d", Get_Speed());
   2029 	mono->Text_Print("X", 16 + (IsLanding?2:0), 12);
   2030 	mono->Text_Print("X", 16 + (IsTakingOff?2:0), 13);
   2031 	FootClass::Debug_Dump(mono);
   2032 }
   2033 #endif
   2034 
   2035 
   2036 /***********************************************************************************************
   2037  * AircraftClass::Active_Click_With -- Handles clicking over specified object.                 *
   2038  *                                                                                             *
   2039  *    This routine is used when the player clicks over the speicifed object. It will assign    *
   2040  *    the appropriate mission to the aircraft.                                                 *
   2041  *                                                                                             *
   2042  * INPUT:   action   -- The action that was nominally determined by the What_Action function.  *
   2043  *                                                                                             *
   2044  *          object   -- The object over which the mouse was clicked.                           *
   2045  *                                                                                             *
   2046  * OUTPUT:  none                                                                               *
   2047  *                                                                                             *
   2048  * WARNINGS:   This routine will alter the game sequence and causes an event packet to be      *
   2049  *             propogated to all connected machines.                                           *
   2050  *                                                                                             *
   2051  * HISTORY:                                                                                    *
   2052  *   06/19/1995 JLB : Created.                                                                 *
   2053  *=============================================================================================*/
   2054 void AircraftClass::Active_Click_With(ActionType action, ObjectClass * object)
   2055 {
   2056 	Validate();
   2057 	switch (action) {
   2058 		case ACTION_ENTER:
   2059 			Player_Assign_Mission(MISSION_ENTER, TARGET_NONE, object->As_Target());
   2060 			break;
   2061 
   2062 		case ACTION_SELF:
   2063 			Player_Assign_Mission(MISSION_UNLOAD, TARGET_NONE, TARGET_NONE);
   2064 			break;
   2065 
   2066 		default:
   2067 			break;
   2068 	}
   2069 	FootClass::Active_Click_With(action, object);
   2070 }
   2071 
   2072 
   2073 /***********************************************************************************************
   2074  * AircraftClass::Active_Click_With -- Handles clicking over specified cell.                   *
   2075  *                                                                                             *
   2076  *    This routine is used when the player clicks the mouse of the specified cell. It will     *
   2077  *    assign the appropriate mission to the aircraft.                                          *
   2078  *                                                                                             *
   2079  * INPUT:   action   -- The action nominally determined by What_Action().                      *
   2080  *                                                                                             *
   2081  *          cell     -- The cell over which the mouse was clicked.                             *
   2082  *                                                                                             *
   2083  * OUTPUT:  none                                                                               *
   2084  *                                                                                             *
   2085  * WARNINGS:   This routine will affect the game sequence and causes an event object to be     *
   2086  *             propogated to all connected machines.                                           *
   2087  *                                                                                             *
   2088  * HISTORY:                                                                                    *
   2089  *   06/19/1995 JLB : Created.                                                                 *
   2090  *=============================================================================================*/
   2091 void AircraftClass::Active_Click_With(ActionType action, CELL cell)
   2092 {
   2093 	Validate();
   2094 #ifdef NEVER
   2095 	switch (action) {
   2096 		case ACTION_MOVE:
   2097 			if (Map[cell].IsVisible) {
   2098 				Player_Assign_Mission(MISSION_MOVE, TARGET_NONE, ::As_Target(cell));
   2099 			}
   2100 			break;
   2101 
   2102 		case ACTION_NOMOVE:
   2103 			break;
   2104 
   2105 		case ACTION_ATTACK:
   2106 			Player_Assign_Mission(MISSION_ATTACK, ::As_Target(cell));
   2107 			break;
   2108 	}
   2109 #endif
   2110 	FootClass::Active_Click_With(action, cell);
   2111 }
   2112 
   2113 
   2114 /***********************************************************************************************
   2115  * AircraftClass::Player_Assign_Mission -- Handles player input to assign a mission.           *
   2116  *                                                                                             *
   2117  *    This routine is called as a result of player input with the intent to change the         *
   2118  *    mission of the aircraft.                                                                 *
   2119  *                                                                                             *
   2120  * INPUT:   mission  -- The mission requested of the aircraft.                                 *
   2121  *                                                                                             *
   2122  *          target   -- The value to assign to the aircraft's targeting computer.              *
   2123  *                                                                                             *
   2124  *          dest.    -- The value to assign to the aircraft's navigation computer.             *
   2125  *                                                                                             *
   2126  * OUTPUT:  none                                                                               *
   2127  *                                                                                             *
   2128  * WARNINGS:   The mission specified will be executed at an indeterminate future game frame.   *
   2129  *             This is controlled by net/modem propogation delay.                              *
   2130  *                                                                                             *
   2131  * HISTORY:                                                                                    *
   2132  *   06/19/1995 JLB : Created.                                                                 *
   2133  *=============================================================================================*/
   2134 void AircraftClass::Player_Assign_Mission(MissionType mission, TARGET target, TARGET destination)
   2135 {
   2136 	Validate();
   2137 	if (AllowVoice) {
   2138 		if (mission == MISSION_ATTACK) {
   2139 			Response_Attack();
   2140 		} else {
   2141 			Response_Move();
   2142 		}
   2143 	}
   2144 	Queue_Mission(As_Target(), mission, target, destination);
   2145 }
   2146 
   2147 
   2148 /***********************************************************************************************
   2149  * AircraftClass::What_Action -- Determines what action to perform.                            *
   2150  *                                                                                             *
   2151  *    This routine is used to determine what action will likely be performed if the mouse      *
   2152  *    were clicked over the object specified. The display system calls this routine to         *
   2153  *    control the mouse shape.                                                                 *
   2154  *                                                                                             *
   2155  * INPUT:   target   -- Pointer to the object that the mouse is currently over.                *
   2156  *                                                                                             *
   2157  * OUTPUT:  Returns with the action that will occur if the mouse were clicked over the         *
   2158  *          object specified.                                                                  *
   2159  *                                                                                             *
   2160  * WARNINGS:   none                                                                            *
   2161  *                                                                                             *
   2162  * HISTORY:                                                                                    *
   2163  *   06/19/1995 JLB : Created.                                                                 *
   2164  *=============================================================================================*/
   2165 ActionType AircraftClass::What_Action(ObjectClass * target) const
   2166 {
   2167 	Validate();
   2168 	ActionType action = FootClass::What_Action(target);
   2169 
   2170 	if (action == ACTION_SELF && (!How_Many() || (Altitude > 0) || IsTethered)) {
   2171 		action = ACTION_NONE;
   2172 	}
   2173 
   2174 	if (action == ACTION_ATTACK && Class->Primary == WEAPON_NONE) {
   2175 		action = ACTION_NONE;
   2176 	}
   2177 
   2178 	//Changed for multiplayer ST - 3/13/2019 5:31PM
   2179 	//if (IsOwnedByPlayer && House->Is_Ally(target) && target->What_Am_I() == RTTI_BUILDING && ((AircraftClass *)this)->Transmit_Message(RADIO_CAN_LOAD, (TechnoClass*)target) == RADIO_ROGER) {
   2180 	if (Is_Owned_By_Player() && House->Is_Ally(target) && target->What_Am_I() == RTTI_BUILDING && ((AircraftClass *)this)->Transmit_Message(RADIO_CAN_LOAD, (TechnoClass*)target) == RADIO_ROGER) {
   2181 		action = ACTION_ENTER;
   2182 	}
   2183 	return(action);
   2184 }
   2185 
   2186 
   2187 /***********************************************************************************************
   2188  * AircraftClass::What_Action -- Determines what action to perform.                            *
   2189  *                                                                                             *
   2190  *    This routine will determine what action would occur if the mouse were clicked over the   *
   2191  *    cell specified. The display system calls this routine to determine what mouse shape      *
   2192  *    to use.                                                                                  *
   2193  *                                                                                             *
   2194  * INPUT:   cell  -- The cell over which the mouse is currently positioned.                    *
   2195  *                                                                                             *
   2196  * OUTPUT:  Returns with the action that will be performed if the mouse were clicked at the    *
   2197  *          specified cell location.                                                           *
   2198  *                                                                                             *
   2199  * WARNINGS:   none                                                                            *
   2200  *                                                                                             *
   2201  * HISTORY:                                                                                    *
   2202  *   06/19/1995 JLB : Created.                                                                 *
   2203  *=============================================================================================*/
   2204 ActionType AircraftClass::What_Action(CELL cell) const
   2205 {
   2206 	Validate();
   2207 	ActionType action = FootClass::What_Action(cell);
   2208 
   2209 	//using function for IsVisible so we have different results for different players - JAS 2019/09/30
   2210 	if ((action == ACTION_MOVE || action == ACTION_ATTACK) && !Map[cell].Is_Visible(PlayerPtr)) {
   2211 		action = ACTION_NOMOVE;
   2212 	}
   2213 
   2214 	if (action == ACTION_ATTACK && Class->Primary == WEAPON_NONE) {
   2215 		action = ACTION_NONE;
   2216 	}
   2217 	return(action);
   2218 }
   2219 
   2220 
   2221 /***********************************************************************************************
   2222  * AircraftClass::Pose_Dir -- Fetches the natural landing facing.                              *
   2223  *                                                                                             *
   2224  *    Use this routine to get the desired facing the aircraft should assume when landing.      *
   2225  *                                                                                             *
   2226  * INPUT:   none                                                                               *
   2227  *                                                                                             *
   2228  * OUTPUT:  Returns with the normal default facing the aircraft should have when landed.       *
   2229  *                                                                                             *
   2230  * WARNINGS:   none                                                                            *
   2231  *                                                                                             *
   2232  * HISTORY:                                                                                    *
   2233  *   06/19/1995 JLB : Created.                                                                 *
   2234  *=============================================================================================*/
   2235 DirType AircraftClass::Pose_Dir(void) const
   2236 {
   2237 	Validate();
   2238 	if (*this == AIRCRAFT_TRANSPORT) {
   2239 		return(DIR_N);
   2240 	}
   2241 	return(DIR_NE);
   2242 }
   2243 
   2244 
   2245 /***********************************************************************************************
   2246  * AircraftClass::Mission_Attack -- Handles the attack mission for aircraft.                   *
   2247  *                                                                                             *
   2248  *    This routine is the state machine that handles the attack mission for aircraft. It will  *
   2249  *    handling homing in on and firing on the target in the aircraft's targeting computer.     *
   2250  *                                                                                             *
   2251  * INPUT:   none                                                                               *
   2252  *                                                                                             *
   2253  * OUTPUT:  Returns with the number of game ticks to pass before this routine must be called   *
   2254  *          again.                                                                             *
   2255  *                                                                                             *
   2256  * WARNINGS:   none                                                                            *
   2257  *                                                                                             *
   2258  * HISTORY:                                                                                    *
   2259  *   06/19/1995 JLB : Created.                                                                 *
   2260  *=============================================================================================*/
   2261 int AircraftClass::Mission_Attack(void)
   2262 {
   2263 	Validate();
   2264 	if (Class->IsFixedWing) {
   2265 		Assign_Mission(MISSION_HUNT);
   2266 		return(1);
   2267 	}
   2268 
   2269 	enum {
   2270 		VALIDATE_AZ,
   2271 		PICK_ATTACK_LOCATION,
   2272 		TAKE_OFF,
   2273 		FLY_TO_POSITION,
   2274 		FIRE_AT_TARGET,
   2275 		FIRE_AT_TARGET2,
   2276 		RETURN_TO_BASE
   2277 	};
   2278 	switch (Status) {
   2279 
   2280 		/*
   2281 		**	Double check target and validate the attack zone.
   2282 		*/
   2283 		case VALIDATE_AZ:
   2284 			if (!Target_Legal(TarCom)) {
   2285 				Status = RETURN_TO_BASE;
   2286 			} else {
   2287 				Status = PICK_ATTACK_LOCATION;
   2288 			}
   2289 			break;
   2290 
   2291 		/*
   2292 		**	Pick a good location to attack from.
   2293 		*/
   2294 		case PICK_ATTACK_LOCATION:
   2295 			if (!Target_Legal(TarCom)) {
   2296 				Status = RETURN_TO_BASE;
   2297 			} else {
   2298 				Assign_Destination(Good_Fire_Location(TarCom));
   2299 				if (Target_Legal(NavCom)) {
   2300 					Status = TAKE_OFF;
   2301 				} else {
   2302 					Status = RETURN_TO_BASE;
   2303 				}
   2304 			}
   2305 			break;
   2306 
   2307 		/*
   2308 		**	Take off (if necessary).
   2309 		*/
   2310 		case TAKE_OFF:
   2311 			if (!Target_Legal(TarCom)) {
   2312 				Status = RETURN_TO_BASE;
   2313 			} else {
   2314 				if (Process_Take_Off()) {
   2315 					Status = FLY_TO_POSITION;
   2316 
   2317 					/*
   2318 					**	Break off radio contact with the helipad it is taking off from.
   2319 					*/
   2320 					if (In_Radio_Contact() && Map[Coord_Cell(Coord)].Cell_Building() == Contact_With_Whom()) {
   2321 						Transmit_Message(RADIO_OVER_OUT);
   2322 					}
   2323 
   2324 					/*
   2325 					**	Start flying toward the destination by skewing at first.
   2326 					**	As the flight progresses, the body will rotate to face
   2327 					**	the direction of travel.
   2328 					*/
   2329 					int diff = SecondaryFacing.Difference(Direction(NavCom));
   2330 					diff = Bound(diff, -128, 128);
   2331 					PrimaryFacing = SecondaryFacing.Current()+diff;
   2332 				}
   2333 				return(1);
   2334 			}
   2335 			break;
   2336 
   2337 		/*
   2338 		**	Fly to attack location.
   2339 		*/
   2340 		case FLY_TO_POSITION:
   2341 			if (Target_Legal(TarCom)) {
   2342 
   2343 				/*
   2344 				**	If the navcom was cleared mysteriously, then try to pick
   2345 				**	a new attack location. This is a likely event if the player
   2346 				**	clicks on a new target while in flight to an existing target.
   2347 				*/
   2348 				if (!Target_Legal(NavCom)) {
   2349 					Status = PICK_ATTACK_LOCATION;
   2350 					return(1);
   2351 				}
   2352 
   2353 				int distance = Process_Fly_To(true);
   2354 
   2355 				if (distance < 0x0200) {
   2356 					SecondaryFacing.Set_Desired(Direction(TarCom));
   2357 
   2358 					if (distance < 0x0010) {
   2359 						Status = FIRE_AT_TARGET;
   2360 						Assign_Destination(TARGET_NONE);
   2361 					}
   2362 				} else {
   2363 					SecondaryFacing.Set_Desired(::Direction(Fire_Coord(0), As_Coord(NavCom)));
   2364 					return(1);
   2365 				}
   2366 			} else {
   2367 				Status = RETURN_TO_BASE;
   2368 			}
   2369 			return(1);
   2370 
   2371 		/*
   2372 		**	Fire at the target.
   2373 		*/
   2374 		case FIRE_AT_TARGET:
   2375 			if (!Target_Legal(TarCom)) {
   2376 				Status = RETURN_TO_BASE;
   2377 				return(1);
   2378 			}
   2379 
   2380 			/*
   2381 			**	Clear second shot flag so fire burst works correctly.
   2382 			*/
   2383 			IsSecondShot = false;
   2384 
   2385 			PrimaryFacing.Set_Desired(Direction(TarCom));
   2386 			SecondaryFacing.Set_Desired(Direction(TarCom));
   2387 			switch (Can_Fire(TarCom, 0)) {
   2388 				case FIRE_CLOAKED:
   2389 					Do_Uncloak();
   2390 					break;
   2391 
   2392 				case FIRE_OK:
   2393 					Fire_At(TarCom, 0);
   2394 					Map[::As_Cell(TarCom)].Incoming(Coord, true);
   2395 					Status = FIRE_AT_TARGET2;
   2396 					break;
   2397 
   2398 				default:
   2399 					if (!Ammo) {
   2400 						Status = RETURN_TO_BASE;
   2401 					} else {
   2402 						Status = FIRE_AT_TARGET2;
   2403 					}
   2404 					break;
   2405 			}
   2406 			return(1);
   2407 
   2408 		/*
   2409 		**	Fire at the target.
   2410 		*/
   2411 		case FIRE_AT_TARGET2:
   2412 			if (!Target_Legal(TarCom)) {
   2413 				Status = RETURN_TO_BASE;
   2414 				return(1);
   2415 			}
   2416 
   2417 			PrimaryFacing.Set_Desired(Direction(TarCom));
   2418 			SecondaryFacing.Set_Desired(Direction(TarCom));
   2419 			switch (Can_Fire(TarCom, 0)) {
   2420 				case FIRE_CLOAKED:
   2421 					Do_Uncloak();
   2422 					break;
   2423 
   2424 				case FIRE_REARM:
   2425 					break;
   2426 
   2427 				case FIRE_OK:
   2428 					Fire_At(TarCom, 0);
   2429 					Map[::As_Cell(TarCom)].Incoming(Coord, true);
   2430 
   2431 					if (Ammo) {
   2432 						Status = PICK_ATTACK_LOCATION;
   2433 					} else {
   2434 						Status = RETURN_TO_BASE;
   2435 					}
   2436 					break;
   2437 
   2438 				default:
   2439 					if (!Ammo) {
   2440 						Status = RETURN_TO_BASE;
   2441 					} else {
   2442 						Status = PICK_ATTACK_LOCATION;
   2443 					}
   2444 					break;
   2445 			}
   2446 			break;
   2447 
   2448 		/*
   2449 		**	Fly back to landing spot.
   2450 		*/
   2451 		case RETURN_TO_BASE:
   2452 			Assign_Destination(TARGET_NONE);
   2453 			Enter_Idle_Mode();
   2454 			break;
   2455 	}
   2456 
   2457 	return(TICKS_PER_SECOND/2);
   2458 }
   2459 
   2460 
   2461 /***********************************************************************************************
   2462  * AircraftClass::New_LZ -- Find a good landing zone.                                          *
   2463  *                                                                                             *
   2464  *    Use this routine to locate a good landing zone that is nearby the location specified.    *
   2465  *    By using this routine it is possible to assign the same landing zone to several          *
   2466  *    aircraft and they will land nearby without conflict.                                     *
   2467  *                                                                                             *
   2468  * INPUT:   oldlz -- Target value of desired landing zone (usually a cell target value).       *
   2469  *                                                                                             *
   2470  * OUTPUT:  Returns with the new good landing zone. It might be the same value passed in.      *
   2471  *                                                                                             *
   2472  * WARNINGS:   The landing zone might be a goodly distance away from the ideal if there is     *
   2473  *             extensive blocking terrain in the vicinity.                                     *
   2474  *                                                                                             *
   2475  * HISTORY:                                                                                    *
   2476  *   06/19/1995 JLB : Created.                                                                 *
   2477  *=============================================================================================*/
   2478 TARGET AircraftClass::New_LZ(TARGET oldlz, bool stable) const
   2479 {
   2480 	Validate();
   2481 	if (Target_Legal(oldlz) && (!Is_LZ_Clear(oldlz) || !Cell_Seems_Ok(As_Cell(oldlz)))) {
   2482 		COORDINATE coord = As_Coord(oldlz);
   2483 
   2484 		/*
   2485 		**	Scan outward in a series of concentric rings up to certain distance
   2486 		**	in cells.
   2487 		*/
   2488 		for (int radius = 0; radius < 16; radius++) {
   2489 			FacingType modifier = stable ? FACING_N : Random_Pick(FACING_N, FACING_NW);
   2490 			CELL lastcell = -1;
   2491 
   2492 			/*
   2493 			**	Perform a radius scan out from the original center location. Try to
   2494 			**	find a cell that is allowed to be a legal LZ.
   2495 			*/
   2496 			for (FacingType facing = FACING_N; facing < FACING_COUNT; facing++) {
   2497 				CELL newcell = Coord_Cell(Coord_Move(coord, Facing_Dir(facing+modifier), radius * ICON_LEPTON_W));
   2498 				if (Map.In_Radar(newcell)) {
   2499 					TARGET newtarget = ::As_Target(newcell);
   2500 
   2501 					if (newcell != lastcell && Is_LZ_Clear(newtarget) && Cell_Seems_Ok(newcell)) {
   2502 						return(newtarget);
   2503 					}
   2504 					lastcell = newcell;
   2505 				}
   2506 			}
   2507 		}
   2508 	}
   2509 	return(oldlz);
   2510 }
   2511 
   2512 
   2513 /***********************************************************************************************
   2514  * AircraftClass::Fire_Coord -- Calculates the point of origin for a bullet.                   *
   2515  *                                                                                             *
   2516  *    This routine is used to find the exact coordinate where the bullet should appear if      *
   2517  *    fired from this object.                                                                  *
   2518  *                                                                                             *
   2519  * INPUT:   which -- Which weapon to consider.                                                 *
   2520  *                                                                                             *
   2521  * OUTPUT:  Returns with the coordinate of where the projectile will appear if fired.          *
   2522  *                                                                                             *
   2523  * WARNINGS:   none                                                                            *
   2524  *                                                                                             *
   2525  * HISTORY:                                                                                    *
   2526  *   06/15/1995 JLB : Created.                                                                 *
   2527  *=============================================================================================*/
   2528 COORDINATE AircraftClass::Fire_Coord(int ) const
   2529 {
   2530 	Validate();
   2531 	return(Coord_Move(Coord_Add(XYP_Coord(0, -Altitude), Coord), SecondaryFacing, 0x040));
   2532 }
   2533 
   2534 
   2535 COORDINATE AircraftClass::Target_Coord(void) const
   2536 {
   2537 	Validate();
   2538 	return(Coord_Add(XYP_Coord(0, -Altitude), Coord));
   2539 }
   2540 
   2541 
   2542 /***********************************************************************************************
   2543  * AircraftClass::Receive_Message -- Handles receipt of radio messages.                        *
   2544  *                                                                                             *
   2545  *    This routine receives all radio messages directed at this aircraft. It is used to handle *
   2546  *    all inter-object coordination. Typically, this would be for transport helicopters and    *
   2547  *    other complex landing operations required of helicopters.                                *
   2548  *                                                                                             *
   2549  * INPUT:   from     -- The source of this radio message.                                      *
   2550  *                                                                                             *
   2551  *          message  -- The message itself.                                                    *
   2552  *                                                                                             *
   2553  *          param    -- An optional parameter that may be used to transfer additional          *
   2554  *                      data.                                                                  *
   2555  *                                                                                             *
   2556  * OUTPUT:  Returns with the radio response from the aircraft.                                 *
   2557  *                                                                                             *
   2558  * WARNINGS:   Some radio messages are handled by the base classes.                            *
   2559  *                                                                                             *
   2560  * HISTORY:                                                                                    *
   2561  *   06/19/1995 JLB : Created.                                                                 *
   2562  *=============================================================================================*/
   2563 RadioMessageType AircraftClass::Receive_Message(RadioClass * from, RadioMessageType message, long & param)
   2564 {
   2565 	Validate();
   2566 	switch (message) {
   2567 
   2568 		case RADIO_PREPARED:
   2569 			if (Target_Legal(TarCom)) return(RADIO_NEGATIVE);
   2570 			if ((Altitude == 0 && Ammo == Class->MaxAmmo) || (Altitude > 0 && Ammo > 0)) return(RADIO_ROGER);
   2571 			return(RADIO_NEGATIVE);
   2572 
   2573 		/*
   2574 		**	Something disasterous has happened to the object in contact with. Fall back
   2575 		**	and regroup. This means that any landing process is immediately aborted.
   2576 		*/
   2577 		case RADIO_RUN_AWAY:
   2578 			if (IsLanding) {
   2579 				IsLanding = false;
   2580 				IsTakingOff = true;
   2581 			}
   2582 			Scatter(0, true);
   2583 			break;
   2584 
   2585 		/*
   2586 		**	The ground control requests that this specified landing spot be used.
   2587 		*/
   2588 		case RADIO_MOVE_HERE:
   2589 			FootClass::Receive_Message(from, message, param);
   2590 			if (Is_Target_Building(param)) {
   2591 				if (Transmit_Message(RADIO_CAN_LOAD, As_Techno(param)) != RADIO_ROGER) {
   2592 					return(RADIO_NEGATIVE);
   2593 				}
   2594 				Assign_Mission(MISSION_ENTER);
   2595 				Assign_Destination((TARGET)param);
   2596 			} else {
   2597 				Assign_Mission(MISSION_MOVE);
   2598 				Assign_Destination((TARGET)param);
   2599 			}
   2600 			Commence();
   2601 			return(RADIO_ROGER);
   2602 
   2603 		/*
   2604 		**	Ground control is requesting if the aircraft requires navigation direction.
   2605 		*/
   2606 		case RADIO_NEED_TO_MOVE:
   2607 			FootClass::Receive_Message(from, message, param);
   2608 			if (!Target_Legal(NavCom) && !IsTakingOff && !IsLanding) {
   2609 				return(RADIO_ROGER);
   2610 			}
   2611 			return(RADIO_NEGATIVE);
   2612 
   2613 		/*
   2614 		**	This message is sent by the passenger when it determines that it has
   2615 		**	entered the transport.
   2616 		*/
   2617 		case RADIO_IM_IN:
   2618 			if (How_Many() == Class->Max_Passengers()) {
   2619 				Close_Door(5, 4);
   2620 			}
   2621 
   2622 			/*
   2623 			**	If a civilian has entered the transport, then the transport will immediately
   2624 			**	fly off the map.
   2625 			*/
   2626 			if (from->What_Am_I() == RTTI_INFANTRY && ((InfantryClass *)from)->Class->IsCivilian && !((InfantryClass *)from)->IsTechnician) {
   2627 				Assign_Mission(MISSION_RETREAT);
   2628 			}
   2629 			return(RADIO_ATTACH);
   2630 
   2631 		/*
   2632 		**	Docking maintenance message received. Check to see if new orders should be given
   2633 		**	to the impatient unit.
   2634 		*/
   2635 		case RADIO_DOCKING:
   2636 			if (Class->IsTransporter && How_Many() < Class->Max_Passengers()) {
   2637 				FootClass::Receive_Message(from, message, param);
   2638 
   2639 				if (!IsTethered && !IsLanding && !IsTakingOff && Altitude == 0) {
   2640 
   2641 					Open_Door(5, 4);
   2642 
   2643 					/*
   2644 					**	If the potential passenger needs someplace to go, then figure out a good
   2645 					**	spot and tell it to go.
   2646 					*/
   2647 					if (Transmit_Message(RADIO_NEED_TO_MOVE, from) == RADIO_ROGER) {
   2648 						CELL cell;
   2649 						DirType dir = Desired_Load_Dir(from, cell);
   2650 
   2651 						/*
   2652 						**	If no adjacent free cells are detected, then passenger loading
   2653 						**	cannot occur. Break radio contact.
   2654 						*/
   2655 						if (cell == 0) {
   2656 							Transmit_Message(RADIO_OVER_OUT, from);
   2657 						} else {
   2658 							param = (long)::As_Target(cell);
   2659 
   2660 							/*
   2661 							**	Tell the potential passenger where it should go. If the passenger is
   2662 							**	already at the staging location, then tell it to move onto the transport
   2663 							**	directly.
   2664 							*/
   2665 							if (Transmit_Message(RADIO_MOVE_HERE, param, from) == RADIO_YEA_NOW_WHAT) {
   2666 								param = (long)As_Target();
   2667 								Transmit_Message(RADIO_TETHER);
   2668 								if (Transmit_Message(RADIO_MOVE_HERE, param, from) != RADIO_ROGER) {
   2669 									Transmit_Message(RADIO_OVER_OUT, from);
   2670 								} else {
   2671 									Contact_With_Whom()->Unselect();
   2672 								}
   2673 							}
   2674 						}
   2675 					}
   2676 				}
   2677 				return(RADIO_ROGER);
   2678 			}
   2679 			break;
   2680 
   2681 		/*
   2682 		**	Asks if the passenger can load on this transport.
   2683 		*/
   2684 		case RADIO_CAN_LOAD:
   2685 			if (!In_Radio_Contact() && Class->IsTransporter && How_Many() < Class->Max_Passengers() && from && House->Class->House == from->Owner() && Altitude == 0) {
   2686 				return(RADIO_ROGER);
   2687 			}
   2688 			return(RADIO_NEGATIVE);
   2689 	}
   2690 
   2691 	/*
   2692 	**	Let the base class take over processing this message.
   2693 	*/
   2694 	return(FootClass::Receive_Message(from, message, param));
   2695 }
   2696 
   2697 
   2698 /***********************************************************************************************
   2699  * AircraftClass::Desired_Load_Dir -- Determines where passengers should line up.              *
   2700  *                                                                                             *
   2701  *    This routine is used by the transport helicopter to determine the location where the     *
   2702  *    infantry passengers should line up before loading.                                       *
   2703  *                                                                                             *
   2704  * INPUT:   object   -- The object that is trying to load up on this transport.                *
   2705  *                                                                                             *
   2706  *                   -- Reference to the cell that the passengers should move to before the    *
   2707  *                      actual load process may begin.                                         *
   2708  *                                                                                             *
   2709  * OUTPUT:  Returns with the direction that the helicopter should face for the load operation. *
   2710  *                                                                                             *
   2711  * WARNINGS:   none                                                                            *
   2712  *                                                                                             *
   2713  * HISTORY:                                                                                    *
   2714  *   06/12/1995 JLB : Created.                                                                 *
   2715  *   07/30/1995 JLB : Revamped to scan all adjacent cells.                                     *
   2716  *=============================================================================================*/
   2717 DirType AircraftClass::Desired_Load_Dir(ObjectClass * object, CELL & moveto) const
   2718 {
   2719 	Validate();
   2720 	CELL center = Coord_Cell(Center_Coord());
   2721 	for (int sweep = FACING_N; sweep < FACING_S; sweep++) {
   2722 		moveto = Adjacent_Cell(center, FACING_S+sweep);
   2723 		if (Map.In_Radar(moveto) && (Coord_Cell(object->Center_Coord()) == moveto || Map[moveto].Is_Generally_Clear())) return(DIR_N);
   2724 
   2725 		moveto = Adjacent_Cell(center, FACING_S-sweep);
   2726 		if (Map.In_Radar(moveto) && (Coord_Cell(object->Center_Coord()) == moveto || Map[moveto].Is_Generally_Clear())) return(DIR_N);
   2727 	}
   2728 	return(DIR_N);
   2729 }
   2730 
   2731 
   2732 /***********************************************************************************************
   2733  * AircraftClass::Process_Take_Off -- State machine support for taking off.                    *
   2734  *                                                                                             *
   2735  *    This routine is used by the main game state machine processor. This utility routine      *
   2736  *    handles a helicopter as it transitions from landed to flying state.                      *
   2737  *                                                                                             *
   2738  * INPUT:   none                                                                               *
   2739  *                                                                                             *
   2740  * OUTPUT:  Has the helicopter reached flight level now?                                       *
   2741  *                                                                                             *
   2742  * WARNINGS:   none                                                                            *
   2743  *                                                                                             *
   2744  * HISTORY:                                                                                    *
   2745  *   06/12/1995 JLB : Created.                                                                 *
   2746  *=============================================================================================*/
   2747 bool AircraftClass::Process_Take_Off(void)
   2748 {
   2749 	Validate();
   2750 	IsLanding = false;
   2751 	IsTakingOff = true;
   2752 	switch (Altitude) {
   2753 		case 0:
   2754 			Close_Door(5, 4);
   2755 			PrimaryFacing = SecondaryFacing;
   2756 			break;
   2757 
   2758 		case FLIGHT_LEVEL/2:
   2759 			PrimaryFacing.Set_Desired(Direction(NavCom));
   2760 			break;
   2761 
   2762 		case FLIGHT_LEVEL-(FLIGHT_LEVEL/3):
   2763 			SecondaryFacing.Set_Desired(PrimaryFacing.Desired());
   2764 			Set_Speed(0x20);
   2765 			break;
   2766 
   2767 		case FLIGHT_LEVEL-(FLIGHT_LEVEL/5):
   2768 			Set_Speed(0x40);
   2769 			break;
   2770 
   2771 		case FLIGHT_LEVEL:
   2772 			Set_Speed(0xFF);
   2773 			IsTakingOff = false;
   2774 			return(true);
   2775 	}
   2776 	return(false);
   2777 }
   2778 
   2779 
   2780 /***********************************************************************************************
   2781  * AircraftClass::Process_Landing -- Landing process state machine handler.                    *
   2782  *                                                                                             *
   2783  *    This is a support routine that is called by the main state machine routines. This        *
   2784  *    routine is responsible for handling the helicopter as it transitions from flight to      *
   2785  *    landing.                                                                                 *
   2786  *                                                                                             *
   2787  * INPUT:   none                                                                               *
   2788  *                                                                                             *
   2789  * OUTPUT:  Has the helicopter completely landed now?                                          *
   2790  *                                                                                             *
   2791  * WARNINGS:   none                                                                            *
   2792  *                                                                                             *
   2793  * HISTORY:                                                                                    *
   2794  *   06/12/1995 JLB : Created.                                                                 *
   2795  *=============================================================================================*/
   2796 bool AircraftClass::Process_Landing(void)
   2797 {
   2798 	Validate();
   2799 	IsTakingOff = false;
   2800 	IsLanding = true;
   2801 	switch (Altitude) {
   2802 		case 0:
   2803 			IsLanding = false;
   2804 			return(true);
   2805 
   2806 		case FLIGHT_LEVEL/2:
   2807 			Set_Speed(0);
   2808 			break;
   2809 
   2810 		case FLIGHT_LEVEL:
   2811 			break;
   2812 	}
   2813 	return(false);
   2814 }
   2815 
   2816 
   2817 /***********************************************************************************************
   2818  * AircraftClass::Can_Enter_Cell -- Determines if the aircraft can land at this location.      *
   2819  *                                                                                             *
   2820  *    This routine is used when the passability of a cell needs to be determined. This is      *
   2821  *    necessary when scanning for a location that the aircraft can land.                       *
   2822  *                                                                                             *
   2823  * INPUT:   cell  -- The cell location to check for landing.                                   *
   2824  *                                                                                             *
   2825  * OUTPUT:  Returns a value indicating if the cell is a legal landing spot or not.             *
   2826  *                                                                                             *
   2827  * WARNINGS:   none                                                                            *
   2828  *                                                                                             *
   2829  * HISTORY:                                                                                    *
   2830  *   06/12/1995 JLB : Created.                                                                 *
   2831  *=============================================================================================*/
   2832 MoveType AircraftClass::Can_Enter_Cell(CELL cell, FacingType ) const
   2833 {
   2834 	Validate();
   2835 	if (!Map.In_Radar(cell)) return(MOVE_NO);
   2836 
   2837 	CellClass * cellptr = &Map[cell];
   2838 
   2839 	if (!cellptr->Is_Generally_Clear(true)) return(MOVE_NO);
   2840 
   2841 	if (GameToPlay == GAME_NORMAL && IsOwnedByPlayer && !cellptr->Is_Visible(PlayerPtr)) return(MOVE_NO);
   2842 
   2843 	return(MOVE_OK);
   2844 }
   2845 
   2846 
   2847 /***********************************************************************************************
   2848  * AircraftClass::Good_Fire_Location -- Searches for and finds a good spot to fire from.       *
   2849  *                                                                                             *
   2850  *    Given the specified target, this routine will locate a good spot for the aircraft to     *
   2851  *    fire at the target.                                                                      *
   2852  *                                                                                             *
   2853  * INPUT:   target   -- The target that is desired to be attacked.                             *
   2854  *                                                                                             *
   2855  * OUTPUT:  Returns with the target location of the place that firing should be made from.     *
   2856  *                                                                                             *
   2857  * WARNINGS:   none                                                                            *
   2858  *                                                                                             *
   2859  * HISTORY:                                                                                    *
   2860  *   06/12/1995 JLB : Created.                                                                 *
   2861  *   06/14/1995 JLB : Finer resolution on ring scan.                                           *
   2862  *=============================================================================================*/
   2863 TARGET AircraftClass::Good_Fire_Location(TARGET target) const
   2864 {
   2865 	Validate();
   2866 	if (Target_Legal(target)) {
   2867 		int range = Weapon_Range(0);
   2868 		COORDINATE tcoord = As_Coord(target);
   2869 		CELL bestcell = 0;
   2870 		CELL best2cell = 0;
   2871 		int bestval = -1;
   2872 		int best2val = -1;
   2873 
   2874 		for (int r = range-0x0180; r > 0x0180; r -= 0x0100) {
   2875 			for (int face = 0; face < 255; face += 16) {
   2876 				COORDINATE newcoord = Coord_Move(tcoord, (DirType)face, r);
   2877 				CELL newcell = Coord_Cell(newcoord);
   2878 				CELL actualcell = Coord_Cell(Coord_Sub(newcoord, XYPixel_Coord(0, FLIGHT_LEVEL)));
   2879 
   2880 				if (Map.In_Radar(actualcell) && (GameToPlay != GAME_NORMAL || Map[newcell].Is_Visible(PlayerPtr)) && Cell_Seems_Ok(newcell, true)) {
   2881 					int dist = Distance(newcoord);
   2882 					if (bestval == -1 || dist < bestval) {
   2883 						best2val = bestval;
   2884 						best2cell = bestcell;
   2885 						bestval = dist;
   2886 						bestcell = newcell;
   2887 					}
   2888 				}
   2889 			}
   2890 			if (bestval != -1) break;
   2891 		}
   2892 
   2893 		if (best2val == -1) {
   2894 			best2cell = bestcell;
   2895 		}
   2896 
   2897 		/*
   2898 		**	If it found a good firing location, then return this location as
   2899 		**	a target value.
   2900 		*/
   2901 		if (bestval != -1) {
   2902 			if (Random_Pick(0, 1) == 0) {
   2903 				return(::As_Target(bestcell));
   2904 			} else {
   2905 				return(::As_Target(best2cell));
   2906 			}
   2907 		}
   2908 	}
   2909 	return(TARGET_NONE);
   2910 }
   2911 
   2912 
   2913 /***********************************************************************************************
   2914  * AircraftClass::Cell_Seems_Ok -- Checks to see if a cell is good to enter.                   *
   2915  *                                                                                             *
   2916  *    This routine examines the navigation computers of other aircraft in order to see if the  *
   2917  *    specified cell is safe to fly to. The intent of this routine is to avoid unneccessary    *
   2918  *    mid-air collisions.                                                                      *
   2919  *                                                                                             *
   2920  * INPUT:   cell     -- The cell to examine for clear airspace.                                *
   2921  *                                                                                             *
   2922  *          strict   -- Should the scan consider the aircraft, that is making this check, a    *
   2923  *                      blocking aircraft. Typically, the aircraft itself is not considered    *
   2924  *                      a blockage -- an aircraft can always exist where it is currently       *
   2925  *                      located. A strict check is useful for helicopters that need to move    *
   2926  *                      around at the slightest provocation.                                   *
   2927  *                                                                                             *
   2928  * OUTPUT:  Is the specified cell free from airspace conflicts?                                *
   2929  *                                                                                             *
   2930  * WARNINGS:   none                                                                            *
   2931  *                                                                                             *
   2932  * HISTORY:                                                                                    *
   2933  *   06/19/1995 JLB : Created.                                                                 *
   2934  *=============================================================================================*/
   2935 bool AircraftClass::Cell_Seems_Ok(CELL cell, bool strict) const
   2936 {
   2937 	Validate();
   2938 	/*
   2939 	**	Make sure that no other aircraft are heading to the selected location. If they
   2940 	**	are, then don't consider the location as valid.
   2941 	*/
   2942 	TARGET astarget = ::As_Target(cell);
   2943 	bool ok = true;
   2944 	for (int index = 0; index < Aircraft.Count(); index++) {
   2945 		AircraftClass * air = Aircraft.Ptr(index);
   2946 		if (air && (strict || air != this) && !air->IsInLimbo) {
   2947 			if (Coord_Cell(air->Coord) == cell || air->NavCom == astarget) {
   2948 				return(false);
   2949 			}
   2950 		}
   2951 	}
   2952 	return(true);
   2953 }
   2954 
   2955 
   2956 /***********************************************************************************************
   2957  * AircraftClass::Pip_Count -- Returns the number of "objects" in aircraft.                    *
   2958  *                                                                                             *
   2959  *    This routine is used by the render logic to draw the little container "pips". This       *
   2960  *    corresponds to the number of passengers for a transport helicopter or the number of      *
   2961  *    shots remaining for an attack helicopter.                                                *
   2962  *                                                                                             *
   2963  * INPUT:   none                                                                               *
   2964  *                                                                                             *
   2965  * OUTPUT:  Returns with the number of "pips" to render on the aircraft.                       *
   2966  *                                                                                             *
   2967  * WARNINGS:   none                                                                            *
   2968  *                                                                                             *
   2969  * HISTORY:                                                                                    *
   2970  *   06/11/1995 JLB : Created.                                                                 *
   2971  *=============================================================================================*/
   2972 int AircraftClass::Pip_Count(void) const
   2973 {
   2974 	Validate();
   2975 	int retval = 0;
   2976 
   2977 	if (Class->IsTransporter) {
   2978 		retval = How_Many();
   2979 	} else {
   2980 		if (Ammo) {
   2981 			retval = Cardinal_To_Fixed(Class->MaxAmmo, Ammo);
   2982 			retval = Fixed_To_Cardinal(Class->Max_Pips(), retval);
   2983 			if (!retval) retval = 1;
   2984 		}
   2985 	}
   2986 	return(retval);
   2987 }
   2988 
   2989 
   2990 /***********************************************************************************************
   2991  * AircraftClass::Mission_Enter -- Control aircraft to fly to the helipad or repair center.    *
   2992  *                                                                                             *
   2993  *    This routine is used when the aircraft needs to fly for either rearming or repairing.    *
   2994  *    It tries to establish contact with the support building. Once contact is established     *
   2995  *    the ground controller takes care of commanding the aircraft.                             *
   2996  *                                                                                             *
   2997  * INPUT:   none                                                                               *
   2998  *                                                                                             *
   2999  * OUTPUT:  Returns with the delay before this routine should be called again.                 *
   3000  *                                                                                             *
   3001  * WARNINGS:   none                                                                            *
   3002  *                                                                                             *
   3003  * HISTORY:                                                                                    *
   3004  *   06/12/1995 JLB : Created.                                                                 *
   3005  *   07/04/1995 JLB : Ground controller gives orders.                                          *
   3006  *=============================================================================================*/
   3007 int AircraftClass::Mission_Enter(void)
   3008 {
   3009 	Validate();
   3010 	enum {
   3011 		INITIAL,
   3012 		TAKEOFF,
   3013 		ALTITUDE,
   3014 		TRAVEL,
   3015 		LANDING
   3016 	};
   3017 	switch (Status) {
   3018 		case INITIAL:
   3019 			if (Altitude < FLIGHT_LEVEL || IsLanding) {
   3020 				Status = TAKEOFF;
   3021 			} else {
   3022 				Status = ALTITUDE;
   3023 			}
   3024 			break;
   3025 
   3026 		case TAKEOFF:
   3027 			if (Process_Take_Off()) {
   3028 				/*
   3029 				**	After takeoff is complete, break radio contact with any helipad that this
   3030 				**	helicopter is taking off from.
   3031 				*/
   3032 				if (In_Radio_Contact() && Map[Coord_Cell(Coord)].Cell_Building() == Contact_With_Whom()) {
   3033 					Transmit_Message(RADIO_OVER_OUT);
   3034 				}
   3035 				Status = ALTITUDE;
   3036 			}
   3037 			break;
   3038 
   3039 		case ALTITUDE:
   3040 			/*
   3041 			**	Establish radio contact with the building this helicopter is trying
   3042 			**	to land at.
   3043 			*/
   3044 			if (In_Radio_Contact()) {
   3045 				Status = TRAVEL;
   3046 			} else {
   3047 				TechnoClass * tech = As_Techno(NavCom);
   3048 				if (tech && Transmit_Message(RADIO_CAN_LOAD, tech) == RADIO_ROGER) {
   3049 					Transmit_Message(RADIO_HELLO, tech);
   3050 					Transmit_Message(RADIO_DOCKING);
   3051 					Status = TRAVEL;
   3052 				} else {
   3053 					Assign_Destination(TARGET_NONE);
   3054 					Enter_Idle_Mode();
   3055 				}
   3056 			}
   3057 			break;
   3058 
   3059 		case TRAVEL:
   3060 			Transmit_Message(RADIO_DOCKING);
   3061 			if (!In_Radio_Contact()) {
   3062 				Assign_Destination(TARGET_NONE);
   3063 				Enter_Idle_Mode();
   3064 			} else {
   3065 				int distance = Process_Fly_To(true);
   3066 
   3067 				if (distance < 0x0080) {
   3068 					if (Target_Legal(TarCom)) {
   3069 						SecondaryFacing.Set_Desired(Direction(TarCom));
   3070 					} else {
   3071 						SecondaryFacing.Set_Desired(Pose_Dir());
   3072 					}
   3073 
   3074 					if (distance < 0x0010) {
   3075 						Status = LANDING;
   3076 					}
   3077 					break;
   3078 				} else {
   3079 					SecondaryFacing.Set_Desired(::Direction(Fire_Coord(0), As_Coord(NavCom)));
   3080 				}
   3081 				return(3);
   3082 			}
   3083 			break;
   3084 
   3085 		case LANDING:
   3086 			if (IsTakingOff) {
   3087 				Assign_Destination(TARGET_NONE);
   3088 				Enter_Idle_Mode();
   3089 			}
   3090 			if (Process_Landing()) {
   3091 				switch (Transmit_Message(RADIO_IM_IN)) {
   3092 					case RADIO_ROGER:
   3093 						Assign_Mission(MISSION_GUARD);
   3094 						break;
   3095 
   3096 					case RADIO_ATTACH:
   3097 						Limbo();
   3098 						Contact_With_Whom()->Attach(this);
   3099 						break;
   3100 
   3101 					default:
   3102 						Enter_Idle_Mode();
   3103 				}
   3104 			}
   3105 			break;
   3106 	}
   3107 	return(1);
   3108 }
   3109 
   3110 
   3111 /***********************************************************************************************
   3112  * AircraftClass::Good_LZ -- Locates a good spot ot land.                                      *
   3113  *                                                                                             *
   3114  *    This routine is used when helicopters need a place to land, but there are no obvious     *
   3115  *    spots (i.e., helipad) available. It will try to land near a friendly helipad or friendly *
   3116  *    building if there are no helipads anywhere. In the event that there are no friendly      *
   3117  *    buildings anywhere on the map, then just land right where it is flying.                  *
   3118  *                                                                                             *
   3119  * INPUT:   none                                                                               *
   3120  *                                                                                             *
   3121  * OUTPUT:  Returns with the target location where this aircraft should land. This value may   *
   3122  *          not be a clear cell, but the normal landing logic will resolve that problem.       *
   3123  *                                                                                             *
   3124  * WARNINGS:   none                                                                            *
   3125  *                                                                                             *
   3126  * HISTORY:                                                                                    *
   3127  *   06/12/1995 JLB : Created.                                                                 *
   3128  *=============================================================================================*/
   3129 TARGET AircraftClass::Good_LZ(void) const
   3130 {
   3131 	Validate();
   3132 	/*
   3133 	**	Scan through all of the buildings and try to land near
   3134 	**	the helipad (if there is one) or the nearest friendly building.
   3135 	*/
   3136 	CELL bestcell;
   3137 	int bestdist = -1;
   3138 	for (int index = 0; index < Buildings.Count(); index++) {
   3139 		BuildingClass * building = Buildings.Ptr(index);
   3140 
   3141 		if (building && !building->IsInLimbo && building->House == House) {
   3142 			int dist = Distance(building);
   3143 			if (*building == STRUCT_HELIPAD) {
   3144 				dist /= 4;
   3145 			}
   3146 			if (bestdist == -1 || dist < bestdist) {
   3147 				bestdist = dist;
   3148 				bestcell = Coord_Cell(building->Center_Coord());
   3149 			}
   3150 		}
   3151 	}
   3152 
   3153 	/*
   3154 	**	Return with the suitable location if one was found.
   3155 	*/
   3156 	if (bestdist != -1) {
   3157 		return(::As_Target(bestcell));
   3158 	}
   3159 
   3160 	/*
   3161 	**	No good location was found. Just try to land here.
   3162 	*/
   3163 	return(::As_Target(Coord_Cell(Coord)));
   3164 }
   3165 
   3166 
   3167 /***********************************************************************************************
   3168  * AircraftClass::Set_Speed -- Sets the speed for the aircraft.                                *
   3169  *                                                                                             *
   3170  *    This routine will set the speed for the aircraft. The speed is specified as a fraction   *
   3171  *    of full speed.                                                                           *
   3172  *                                                                                             *
   3173  * INPUT:   speed -- The fixed point fractional speed setting. 0x00 is stopped, 0xFF is full   *
   3174  *                   speed.                                                                    *
   3175  *                                                                                             *
   3176  * OUTPUT:  none                                                                               *
   3177  *                                                                                             *
   3178  * WARNINGS:   none                                                                            *
   3179  *                                                                                             *
   3180  * HISTORY:                                                                                    *
   3181  *   06/19/1995 JLB : Created.                                                                 *
   3182  *=============================================================================================*/
   3183 void AircraftClass::Set_Speed(int speed)
   3184 {
   3185 	Validate();
   3186 	MPHType sp = MPHType(min((unsigned)(Class->MaxSpeed * House->AirspeedBias), MPH_LIGHT_SPEED));
   3187 	Fly_Speed(speed, sp);
   3188 }
   3189 
   3190 
   3191 /***********************************************************************************************
   3192  * AircraftClass::Fire_Direction -- Determines the direction of fire.                          *
   3193  *                                                                                             *
   3194  *    This routine will determine what direction a projectile would take if it were fired      *
   3195  *    from the aircraft. This is the direction that the aircraft's body is facing.             *
   3196  *                                                                                             *
   3197  * INPUT:   none                                                                               *
   3198  *                                                                                             *
   3199  * OUTPUT:  Returns with the direction of projectile fire.                                     *
   3200  *                                                                                             *
   3201  * WARNINGS:   none                                                                            *
   3202  *                                                                                             *
   3203  * HISTORY:                                                                                    *
   3204  *   06/19/1995 JLB : Created.                                                                 *
   3205  *=============================================================================================*/
   3206 DirType AircraftClass::Fire_Direction(void) const
   3207 {
   3208 	Validate();
   3209 	return(SecondaryFacing.Current());
   3210 }
   3211 
   3212 
   3213 /***********************************************************************************************
   3214  * AircraftClass::~AircraftClass -- Destructor for aircraft object.                            *
   3215  *                                                                                             *
   3216  *    This is the destructor for aircraft. It will limbo the aircraft if it isn't already      *
   3217  *    and also removes the aircraft from any team it may be attached to.                       *
   3218  *                                                                                             *
   3219  * INPUT:   none                                                                               *
   3220  *                                                                                             *
   3221  * OUTPUT:  none                                                                               *
   3222  *                                                                                             *
   3223  * WARNINGS:   none                                                                            *
   3224  *                                                                                             *
   3225  * HISTORY:                                                                                    *
   3226  *   06/24/1995 JLB : Created.                                                                 *
   3227  *=============================================================================================*/
   3228 AircraftClass::~AircraftClass(void)
   3229 {
   3230 	if (GameActive && Class) {
   3231 		
   3232 #ifdef USE_RA_AI
   3233 		//
   3234 		// Added for RA AI in TD. ST - 7/26/2019 9:12AM
   3235 		//
   3236 		House->Tracking_Remove(this);
   3237 #endif
   3238 
   3239 		/*
   3240 		**	If there are any cargo members, delete them.
   3241 		*/
   3242 		while (Is_Something_Attached()) {
   3243 			delete Detach_Object();
   3244 		}
   3245 
   3246 		Limbo();
   3247 	}
   3248 
   3249 	if (GameActive && Class && Team) Team->Remove(this);
   3250 }
   3251 
   3252 
   3253 /***********************************************************************************************
   3254  * AircraftClass::Scatter -- Causes the aircraft to move away a bit.                           *
   3255  *                                                                                             *
   3256  *    This routine will cause the aircraft to move away from its current location and then     *
   3257  *    enter some idle mode. Typically this is called when the aircraft is attacked while on    *
   3258  *    the ground.                                                                              *
   3259  *                                                                                             *
   3260  * INPUT:   none                                                                               *
   3261  *                                                                                             *
   3262  * OUTPUT:  none                                                                               *
   3263  *                                                                                             *
   3264  * WARNINGS:   none                                                                            *
   3265  *                                                                                             *
   3266  * HISTORY:                                                                                    *
   3267  *   07/08/1995 JLB : Created.                                                                 *
   3268  *=============================================================================================*/
   3269 void AircraftClass::Scatter(COORDINATE , bool , bool )
   3270 {
   3271 	Validate();
   3272 	if (IsLanding || Altitude == 0) {
   3273 		IsLanding = false;
   3274 		IsTakingOff = true;
   3275 	}
   3276 	Enter_Idle_Mode();
   3277 }
   3278 
   3279 
   3280 /***********************************************************************************************
   3281  * AircraftClass::Rearm_Delay -- Returns the delay between shots for this aircraft.            *
   3282  *                                                                                             *
   3283  *    Aircraft have a faster rearm delay than their weapon would otherwise indicate. This is   *
   3284  *    necessary to give helicopters a combat edge while still allowing them to share the       *
   3285  *    weapon types used by ground units.                                                       *
   3286  *                                                                                             *
   3287  * INPUT:   second   -- Is this for the second shot? The second shot uses the full rearm       *
   3288  *                      delay. The first shot, if part of a two shot weapon, is given an       *
   3289  *                      abbreviated rearm time.                                                *
   3290  *                                                                                             *
   3291  * OUTPUT:  Returns with the game frames to delay before the next shot can fire.               *
   3292  *                                                                                             *
   3293  * WARNINGS:   none                                                                            *
   3294  *                                                                                             *
   3295  * HISTORY:                                                                                    *
   3296  *   08/10/1995 JLB : Created.                                                                 *
   3297  *=============================================================================================*/
   3298 int AircraftClass::Rearm_Delay(bool second) const
   3299 {
   3300 	Validate();
   3301 	return(FootClass::Rearm_Delay(second)/2);
   3302 }
   3303 
   3304 
   3305 /***********************************************************************************************
   3306  * AircraftClass::Threat_Range -- Returns with a range to scan for targets.                    *
   3307  *                                                                                             *
   3308  *    This routine returns with the distance to scan for targets according to the type of      *
   3309  *    search requested. The search type is typically the weapon (or short) range and vicinity  *
   3310  *    distances. This is used by Guard and Guard Area missions. Aircraft never consider their  *
   3311  *    weapon range in this determination since they can fly so fast, weapon range is           *
   3312  *    meaningless.                                                                             *
   3313  *                                                                                             *
   3314  * INPUT:   control  -- The range control parameter;                                           *
   3315  *                      -1 = range doesn't matter -- return -1 for compatability reasons.      *
   3316  *                      0  = short range scan                                                  *
   3317  *                      1  = long range scan                                                   *
   3318  *                                                                                             *
   3319  * OUTPUT:  Returns with the lepton distance to use for the scan operation.                    *
   3320  *                                                                                             *
   3321  * WARNINGS:   none                                                                            *
   3322  *                                                                                             *
   3323  * HISTORY:                                                                                    *
   3324  *   07/18/1995 JLB : Created.                                                                 *
   3325  *=============================================================================================*/
   3326 int AircraftClass::Threat_Range(int control) const
   3327 {
   3328 	Validate();
   3329 	if (control == -1) return(-1);
   3330 
   3331 	int range = 20 * ICON_LEPTON_W;
   3332 	if (control == 1) {
   3333 		range *= 2;
   3334 	}
   3335 	return(range);
   3336 }
   3337 
   3338 
   3339 /***********************************************************************************************
   3340  * AircraftClass::Mission_Guard -- Handles aircraft in guard mode.                             *
   3341  *                                                                                             *
   3342  *    Aircraft don't like to be in guard mode if in flight. If this situation is detected,     *
   3343  *    then figure out what the aircraft should be doing and go do it.                          *
   3344  *                                                                                             *
   3345  * INPUT:   none                                                                               *
   3346  *                                                                                             *
   3347  * OUTPUT:  Returns with the number of game frames to delay before calling this routine again. *
   3348  *                                                                                             *
   3349  * WARNINGS:   This routine typically calls the normal guard logic for ground units.           *
   3350  *                                                                                             *
   3351  * HISTORY:                                                                                    *
   3352  *   07/18/1995 JLB : Created.                                                                 *
   3353  *=============================================================================================*/
   3354 int AircraftClass::Mission_Guard(void)
   3355 {
   3356 	Validate();
   3357 	if (Altitude == FLIGHT_LEVEL) {
   3358 
   3359 		/*
   3360 		**	If part of a team, then do nothing, since the team
   3361 		**	handler will take care of giving this aircraft a
   3362 		**	mission.
   3363 		*/
   3364 		if (Team) {
   3365 			if (Target_Legal(NavCom)) {
   3366 				Assign_Mission(MISSION_MOVE);
   3367 			}
   3368 			return(TICKS_PER_SECOND);
   3369 		}
   3370 
   3371 		if (Class->Primary == WEAPON_NONE) {
   3372 			Assign_Destination(::As_Target(Coord_Cell(Coord)));
   3373 			Assign_Mission(MISSION_MOVE);
   3374 		} else {
   3375 			Enter_Idle_Mode();
   3376 		}
   3377 		return(1);
   3378 	}
   3379 	if (House->IsHuman) return(TICKS_PER_SECOND);
   3380 
   3381 	/*
   3382 	**	Special case to force the GDI helicopter to be brain dead in the Nod
   3383 	**	mission where it is supposed to be captured.
   3384 	*/
   3385 	if (GameToPlay == GAME_NORMAL && Scenario == 7 && House->Class->House == HOUSE_GOOD) {
   3386 		return(TICKS_PER_SECOND*20);
   3387 	}
   3388 
   3389 	/*
   3390 	**	If the aircraft is very badly damaged, then it will search for a
   3391 	**	repair bay first.
   3392 	*/
   3393 	if (House->Available_Money() >= 100 && Health_Ratio() <= 0x0080) {
   3394 		if (!In_Radio_Contact() ||
   3395 			(Altitude == 0 &&
   3396 				(Contact_With_Whom()->What_Am_I() != RTTI_BUILDING || *((BuildingClass *)Contact_With_Whom()) != STRUCT_REPAIR))) {
   3397 
   3398 
   3399 			BuildingClass * building = Find_Docking_Bay(STRUCT_REPAIR, true);
   3400 			if (building) {
   3401 				Assign_Destination(building->As_Target());
   3402 				Assign_Target(TARGET_NONE);
   3403 				Assign_Mission(MISSION_ENTER);
   3404 				return(1);
   3405 			}
   3406 		}
   3407 	}
   3408 
   3409 	/*
   3410 	**	If the aircraft cannot attack anything because of lack of ammo,
   3411 	**	abort any normal guard logic in order to look for a helipad
   3412 	**	to rearm.
   3413 	*/
   3414 	if (Ammo == 0 && Class->Primary != WEAPON_NONE) {
   3415 		if (!In_Radio_Contact()) {
   3416 			BuildingClass * building = Find_Docking_Bay(STRUCT_HELIPAD, false);
   3417 			if (building) {
   3418 				Assign_Destination(building->As_Target());
   3419 				Assign_Target(TARGET_NONE);
   3420 				Assign_Mission(MISSION_ENTER);
   3421 				return(1);
   3422 			}
   3423 		}
   3424 //		return(TICKS_PER_SECOND*3);
   3425 	}
   3426 
   3427 	/*
   3428 	**	If the aircraft already has a target, then attack it if possible.
   3429 	*/
   3430 	if (Target_Legal(TarCom)) {
   3431 		Assign_Mission(MISSION_ATTACK);
   3432 		return(1);
   3433 	}
   3434 
   3435 	/*
   3436 	**	Transport helicopters don't really do anything but just sit there.
   3437 	*/
   3438 	if (Class->Primary == WEAPON_NONE) {
   3439 		return(TICKS_PER_SECOND*3);
   3440 	}
   3441 
   3442 	/*
   3443 	**	Computer controlled helicopters will defend themselves by bouncing around
   3444 	**	and looking for a free helipad.
   3445 	*/
   3446 	if (Altitude == 0 && !In_Radio_Contact()) {
   3447 		Scatter(0, true);
   3448 		return(TICKS_PER_SECOND*3);
   3449 	}
   3450 
   3451 	return(FootClass::Mission_Guard());
   3452 }
   3453 
   3454 
   3455 /***********************************************************************************************
   3456  * AircraftClass::Mission_Guard_Area -- Handles the aircraft guard area logic.                 *
   3457  *                                                                                             *
   3458  *    This routine handles area guard logic for aircraft. Aircraft require special handling    *
   3459  *    for this mode since they are to guard area only if they are in a position to do so.      *
   3460  *    Otherwise they just defend themselves.                                                   *
   3461  *                                                                                             *
   3462  * INPUT:   none                                                                               *
   3463  *                                                                                             *
   3464  * OUTPUT:  Returns with the number of game frames to delay before calling this routine        *
   3465  *          again.                                                                             *
   3466  *                                                                                             *
   3467  * WARNINGS:   none                                                                            *
   3468  *                                                                                             *
   3469  * HISTORY:                                                                                    *
   3470  *   08/10/1995 JLB : Created.                                                                 *
   3471  *=============================================================================================*/
   3472 int AircraftClass::Mission_Guard_Area(void)
   3473 {
   3474 	Validate();
   3475 	if (Altitude == FLIGHT_LEVEL) {
   3476 		Enter_Idle_Mode();
   3477 		return(1);
   3478 	}
   3479 	if (House->IsHuman) return(TICKS_PER_SECOND);
   3480 
   3481 	if (Altitude == 0 && !In_Radio_Contact()) {
   3482 		Scatter(0, true);
   3483 		return(TICKS_PER_SECOND*3);
   3484 	}
   3485 
   3486 	if (Target_Legal(TarCom)) {
   3487 		Assign_Mission(MISSION_ATTACK);
   3488 		return(1);
   3489 	}
   3490 	return(FootClass::Mission_Guard_Area());
   3491 }
   3492 
   3493 
   3494 /***********************************************************************************************
   3495  * AircraftClass::Response_Attack -- Gives audio response to attack order.                     *
   3496  *                                                                                             *
   3497  *    This routine is used to give an audio response to an attack order.                       *
   3498  *                                                                                             *
   3499  * INPUT:   none                                                                               *
   3500  *                                                                                             *
   3501  * OUTPUT:  none                                                                               *
   3502  *                                                                                             *
   3503  * WARNINGS:   none                                                                            *
   3504  *                                                                                             *
   3505  * HISTORY:                                                                                    *
   3506  *   08/10/1995 JLB : Created.                                                                 *
   3507  *=============================================================================================*/
   3508 void AircraftClass::Response_Attack(void)
   3509 {
   3510 	Validate();
   3511 	static VocType _response[] = {
   3512 		VOC_AFFIRM,
   3513 		VOC_ACKNOWL,
   3514 		VOC_YESSIR,
   3515 		VOC_YESSIR,
   3516 		VOC_YESSIR
   3517 	};
   3518 	VocType response = _response[Sim_Random_Pick(0, (int)(sizeof(_response) / sizeof(_response[0]))-1)];
   3519 	if (AllowVoice) {
   3520 		Sound_Effect(response, 0, -(Aircraft.ID(this)+1));
   3521 	}
   3522 }
   3523 
   3524 
   3525 /***********************************************************************************************
   3526  * AircraftClass::Response_Move -- Gives audio response to move request.                       *
   3527  *                                                                                             *
   3528  *    This routine is used to give an audio response to movement orders.                       *
   3529  *                                                                                             *
   3530  * INPUT:   none                                                                               *
   3531  *                                                                                             *
   3532  * OUTPUT:  none                                                                               *
   3533  *                                                                                             *
   3534  * WARNINGS:   none                                                                            *
   3535  *                                                                                             *
   3536  * HISTORY:                                                                                    *
   3537  *   08/10/1995 JLB : Created.                                                                 *
   3538  *=============================================================================================*/
   3539 void AircraftClass::Response_Move(void)
   3540 {
   3541 	Validate();
   3542 	static VocType _response[] = {
   3543 		VOC_MOVEOUT,
   3544 		VOC_MOVEOUT,
   3545 		VOC_MOVEOUT,
   3546 		VOC_ACKNOWL,
   3547 		VOC_AFFIRM,
   3548 		VOC_AFFIRM
   3549 	};
   3550 	VocType response = _response[Sim_Random_Pick(0, (int)(sizeof(_response) / sizeof(_response[0]))-1)];
   3551 	if (AllowVoice) {
   3552 		Sound_Effect(response, 0, -(Aircraft.ID(this)+1));
   3553 	}
   3554 }
   3555 
   3556 
   3557 /***********************************************************************************************
   3558  * AircraftClass::Response_Select -- Gives audio response when selected.                       *
   3559  *                                                                                             *
   3560  *    This routine is called when an audio response for selection is desired.                  *
   3561  *                                                                                             *
   3562  * INPUT:   none                                                                               *
   3563  *                                                                                             *
   3564  * OUTPUT:  none                                                                               *
   3565  *                                                                                             *
   3566  * WARNINGS:   none                                                                            *
   3567  *                                                                                             *
   3568  * HISTORY:                                                                                    *
   3569  *   08/10/1995 JLB : Created.                                                                 *
   3570  *=============================================================================================*/
   3571 void AircraftClass::Response_Select(void)
   3572 {
   3573 	Validate();
   3574 	static VocType _response[] = {
   3575 		VOC_VEHIC,
   3576 		VOC_UNIT,
   3577 		VOC_YESSIR,
   3578 		VOC_YESSIR,
   3579 		VOC_YESSIR,
   3580 		VOC_AWAIT
   3581 	};
   3582 	VocType response = _response[Sim_Random_Pick(0, (int)(sizeof(_response) / sizeof(_response[0]))-1)];
   3583 	if (AllowVoice) {
   3584 		Sound_Effect(response, 0, -(Aircraft.ID(this)+1));
   3585 	}
   3586 }
   3587 
   3588 void AircraftClass::Set_Reinforcement_Delay(long delay)
   3589 {
   3590 	ReinforcementStart = Frame + delay;
   3591 }