CnC_Remastered_Collection

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

TECHNO.CPP (235466B)


      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\techno.cpv   2.13   02 Aug 1995 17:01:08   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 : BASE.CPP                                                     *
     24  *                                                                                             *
     25  *                   Programmer : Joe L. Bostic                                                *
     26  *                                                                                             *
     27  *                   Start Date : May 8, 1994                                                  *
     28  *                                                                                             *
     29  *                  Last Update : August 23, 1995 [JLB]                                        *
     30  *                                                                                             *
     31  *---------------------------------------------------------------------------------------------*
     32  * Functions:                                                                                  *
     33  *   TechnoClass::AI -- Handles AI processing for techno object.                               *
     34  *   TechnoClass::Assign_Destination -- Assigns movement destination to the object.            *
     35  *   TechnoClass::Assign_Target -- Assigns the targeting computer with specified target.       *
     36  *   TechnoClass::Base_Is_Attacked -- Handle panic response to base being attacked.            *
     37  *   TechnoClass::Can_Fire -- Determines if this techno object can fire.                       *
     38  *   TechnoClass::Can_Player_Fire -- Determines if the player can give this object a fire order*
     39  *   TechnoClass::Can_Player_Move -- Determines if the object can move be moved by player.     *
     40  *   TechnoClass::Can_Repair -- Determines if the object can and should be repaired.           *
     41  *   TechnoClass::Captured -- Handles capturing this object.                                   *
     42  *   TechnoClass::Clicked_As_Target -- Sets the flash count for this techno object.            *
     43  *   TechnoClass::Crew_Type -- Fetches the kind of crew this object contains.                  *
     44  *   TechnoClass::Debug_Dump -- Displays the base class data to the monochrome screen.         *
     45  *   TechnoClass::Desired_Load_Dir -- Fetches loading parameters for this object.              *
     46  *   TechnoClass::Detach -- Handles removal of target from tracking system.                    *
     47  *   TechnoClass::Do_Cloak -- Start the object into cloaking stage.                            *
     48  *   TechnoClass::Do_Shimmer -- Causes this object to shimmer if it is cloaked.                *
     49  *   TechnoClass::Do_Uncloak -- Cause the stealth tank to uncloak.                             *
     50  *   TechnoClass::Draw_It -- Draws the health bar (if necessary).                              *
     51  *   TechnoClass::Draw_Pips -- Draws the transport pips and other techno graphics.             *
     52  *   TechnoClass::Enter_Idle_Mode -- Object enters its default idle condition.                 *
     53  *   TechnoClass::Evaluate_Cell -- Determine the value and object of specified cell.           *
     54  *   TechnoClass::Evaluate_Object -- Determines score value of specified object.               *
     55  *   TechnoClass::Exit_Object -- Causes specified object to leave this object.                 *
     56  *   TechnoClass::Find_Docking_Bay -- Searches for a close docking bay.                        *
     57  *   TechnoClass::Find_Exit_Cell -- Finds an appropriate exit cell for this object.            *
     58  *   TechnoClass::Fire_At -- Fires projectile at target specified.                             *
     59  *   TechnoClass::Fire_Direction -- Fetches the direction projectile fire will take.           *
     60  *   TechnoClass::Get_Ownable -- Fetches the ownable bits for this object.                     *
     61  *   TechnoClass::Greatest_Threat -- Determines best target given search criteria.             *
     62  *   TechnoClass::Hidden -- Returns the object back into the hidden state.                     *
     63  *   TechnoClass::In_Range -- Determines if specified target is within weapon range.           *
     64  *   TechnoClass::In_Range -- Determines if specified target is within weapon range.           *
     65  *   TechnoClass::In_Range -- Determines if the specified coordinate is within range.          *
     66  *   TechnoClass::Is_Techno -- Confirms that this is a TechnoClass object.                     *
     67  *   TechnoClass::Is_Weapon_Equipped -- Determines if this object has a combat weapon.         *
     68  *   TechnoClass::Kill_Cargo -- Destroys any cargo attached to this object.                    *
     69  *   TechnoClass::Mark -- Handles marking of techno objects.                                   *
     70  *   TechnoClass::Nearby_Location -- Radiates outward looking for clear cell nearby.           *
     71  *   TechnoClass::Owner -- Who is the owner of this object?                                    *
     72  *   TechnoClass::Per_Cell_Process -- Handles once-per-cell operations for techno type objects.*
     73  *   TechnoClass::Pip_Count -- Fetches the number of pips to display on this object.           *
     74  *   TechnoClass::Player_Assign_Mission -- Assigns a mission as result of player input.        *
     75  *   TechnoClass::Random_Animate -- Performs some idle animation for the object.               *
     76  *   TechnoClass::Rearm_Delay -- Calculates the delay before firing can occur.                 *
     77  *   TechnoClass::Receive_Message -- Handles inbound message as appropriate.                   *
     78  *   TechnoClass::Record_The_Kill -- Records the death of this object.                         *
     79  *   TechnoClass::Remap_Table -- Fetches the appropriate remap table to use.                   *
     80  *   TechnoClass::Response_Attack -- Handles the voice response when given attack order.       *
     81  *   TechnoClass::Response_Move -- Handles the voice repsonse to a movement request.           *
     82  *   TechnoClass::Response_Select -- Handles the voice response when selected.                 *
     83  *   TechnoClass::Revealed -- Handles revealing an object to the house specified.              *
     84  *   TechnoClass::Risk -- Fetches the risk associated with this object.                        *
     85  *   TechnoClass::Scatter -- Causes the object to scatter to an adjacent cell.                 *
     86  *   TechnoClass::Select -- Selects object and checks to see if can be selected.               *
     87  *   TechnoClass::Set_Mission -- Forced mission set (used by editor).                          *
     88  *   TechnoClass::Stun -- Prepares the object for removal from the game.                       *
     89  *   TechnoClass::Take_Damage -- Records damage assessed to this object.                       *
     90  *   TechnoClass::Target_Something_Nearby -- Handles finding and assigning a nearby target.    *
     91  *   TechnoClass::TechnoClass -- Constructor for techno type objects.                          *
     92  *   TechnoClass::TechnoClass -- Default constructor for techno objects.                       *
     93  *   TechnoClass::Techno_Draw_Object -- General purpose draw object routine.                   *
     94  *   TechnoClass::Threat_Range -- Returns the range to scan based on threat control.           *
     95  *   TechnoClass::Tiberium_Load -- Fetches the current tiberium load percentage.               *
     96  *   TechnoClass::Unlimbo -- Performs unlimbo process for all techno type objects.             *
     97  *   TechnoClass::Value -- Fetches the target value for this object.                           *
     98  *   TechnoClass::Visual_Character -- Determine the visual character of the object.            *
     99  *   TechnoClass::Weapon_Range -- Determines the maximum range for the weapon.                 *
    100  *   TechnoClass::What_Action -- Determines action to perform if cell is clicked on.           *
    101  *   TechnoClass::What_Action -- Determines what action to perform if object is selected.      *
    102  *   TechnoTypeClass::Cost_Of -- Fetches the cost of this object type.                         *
    103  *   TechnoTypeClass::Get_Cameo_Data -- Fetches the cameo image for this object type.          *
    104  *   TechnoTypeClass::Get_Ownable -- Fetches the ownable bits for this object type.            *
    105  *   TechnoTypeClass::Max_Passengers -- Fetches the maximum passengers allowed.                *
    106  *   TechnoTypeClass::Repair_Cost -- Fetches the cost to repair one step.                      *
    107  *   TechnoTypeClass::Repair_Step -- Fetches the health to repair one step.                    *
    108  *   TechnoTypeClass::TechnoTypeClass -- Constructor for techno type objects.                  *
    109  *   TechnoTypeClass::Time_To_Build -- Fetches the time to build this object.                  *
    110  *   TechnoTypeClass::Raw_Cost -- Fetches the raw (base) cost of the object.                   *
    111  *   TechnoClass::Refund_Amount -- Returns with the money to refund if this object is sold.    *
    112  * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
    113 
    114 #include	"function.h"
    115 
    116 
    117 /***************************************************************************
    118 **	Cloaking control values.
    119 */
    120 #define	MAX_UNCLOAK_STAGE		38
    121 #define	UNCLOAK_VIS_TIME		(1*TICKS_PER_SECOND)
    122 
    123 //Added for getting the input for special character keys from the client 
    124 // - 6/26/2019 JAS 
    125 extern bool DLL_Export_Get_Input_Key_State(KeyNumType key);
    126 
    127 
    128 /***************************************************************************
    129 **	Which shape to use depending on which facing is controlled by these arrays.
    130 */
    131 int const TechnoClass::BodyShape[32] = {0,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,1};
    132 
    133 
    134 /***********************************************************************************************
    135  * TechnoTypeClass::TechnoTypeClass -- Constructor for techno type objects.                    *
    136  *                                                                                             *
    137  *    This is the normal constructor for techno type objects. It is called in the process of   *
    138  *    constructing all the object type (constant) data for the various techno type objects.    *
    139  *                                                                                             *
    140  * INPUT:   see below...                                                                       *
    141  *                                                                                             *
    142  * OUTPUT:  none                                                                               *
    143  *                                                                                             *
    144  * WARNINGS:   none                                                                            *
    145  *                                                                                             *
    146  * HISTORY:                                                                                    *
    147  *   03/19/1995 JLB : Created.                                                                 *
    148  *=============================================================================================*/
    149 TechnoTypeClass::TechnoTypeClass(
    150 		int name,
    151 		char const *ininame,
    152 		unsigned char level,
    153 		long pre,
    154 		bool is_leader,
    155 		bool is_scanner,
    156 		bool is_nominal,
    157 		bool is_transporter,
    158 		bool is_flammable,
    159 		bool is_crushable,
    160 		bool is_stealthy,
    161 		bool is_selectable,
    162 		bool is_legal_target,
    163 		bool is_insignificant,
    164 		bool is_immune,
    165 		bool is_theater,
    166 		bool is_twoshooter,
    167 		bool is_turret_equipped,
    168 		bool is_repairable,
    169 		bool is_buildable,
    170 		bool is_crew,
    171 		int ammo,
    172 		unsigned short strength,
    173 		MPHType maxspeed,
    174 		int sightrange,
    175 		int cost,
    176 		int scenario,
    177 		int risk,
    178 		int reward,
    179 		int ownable,
    180 		WeaponType primary,
    181 		WeaponType secondary,
    182 		ArmorType armor) :
    183 	ObjectTypeClass(	true,
    184 							is_flammable,
    185 							is_crushable,
    186 							is_stealthy,
    187 							is_selectable,
    188 							is_legal_target,
    189 							is_insignificant,
    190 							is_immune,
    191 							name,
    192 							ininame,
    193 							armor,
    194 							strength)
    195 {
    196 	Level = level;
    197 	Pre = pre;
    198 	MaxAmmo = ammo;
    199 	MaxSpeed = maxspeed;
    200 	CameoData = NULL;
    201 	Primary = primary,
    202 	Secondary = secondary,
    203 	Cost = cost;
    204 	IsLeader = is_leader;
    205 	IsScanner = is_scanner;
    206 	IsTransporter = is_transporter;
    207 	IsTwoShooter = is_twoshooter;
    208 	IsBuildable = is_buildable;
    209 	IsCrew = is_crew;
    210 	IsTheater = is_theater;
    211 	IsRepairable = is_repairable;
    212 	IsTurretEquipped= is_turret_equipped;
    213 	IsNominal = is_nominal;
    214 	Ownable = ownable;
    215 	Reward = reward;
    216 	Scenario = scenario;
    217 	SightRange = sightrange;
    218 
    219 	/*
    220 	** Units risk value is based on the type of weapon he has and the
    221 	** rate of fire it shoots at.
    222 	*/
    223 	risk = risk;
    224 	Risk = 0;
    225 	if (primary != WEAPON_NONE) {
    226 		Risk = (Weapons[primary].Attack * (Weapons[primary].Range >> 4)) / Weapons[primary].ROF;
    227 	}
    228 }
    229 
    230 
    231 /***********************************************************************************************
    232  * TechnoTypeClass::Raw_Cost -- Fetches the raw (base) cost of the object.                     *
    233  *                                                                                             *
    234  *    This routine is used to find the underlying cost for this object. The underlying cost    *
    235  *    does not include any free items that normally come with the object when purchased        *
    236  *    directly. Example: The raw cost of a refinery is the normal cost minus the cost of a     *
    237  *    harvester.                                                                               *
    238  *                                                                                             *
    239  * INPUT:   none                                                                               *
    240  *                                                                                             *
    241  * OUTPUT:  Returns with the credit cost of the base object type.                              *
    242  *                                                                                             *
    243  * WARNINGS:   none                                                                            *
    244  *                                                                                             *
    245  * HISTORY:                                                                                    *
    246  *   08/13/1995 JLB : Created.                                                                 *
    247  *=============================================================================================*/
    248 int TechnoTypeClass::Raw_Cost(void) const
    249 {
    250 	return(Cost);
    251 }
    252 
    253 
    254 /***********************************************************************************************
    255  * TechnoTypeClass::Get_Ownable -- Fetches the ownable bits for this object type.              *
    256  *                                                                                             *
    257  *    This routine will return the ownable bits for this object type. The ownable bits are     *
    258  *    a bitflag composite of the houses that can own (build) this object type.                 *
    259  *                                                                                             *
    260  * INPUT:   none                                                                               *
    261  *                                                                                             *
    262  * OUTPUT:  Returns with the ownable bits for this object type.                                *
    263  *                                                                                             *
    264  * WARNINGS:   none                                                                            *
    265  *                                                                                             *
    266  * HISTORY:                                                                                    *
    267  *   07/29/1995 JLB : Created.                                                                 *
    268  *=============================================================================================*/
    269 unsigned short TechnoTypeClass::Get_Ownable(void) const
    270 {
    271 	return(Ownable);
    272 }
    273 
    274 
    275 /***********************************************************************************************
    276  * TechnoTypeClass::Time_To_Build -- Fetches the time to build this object.                    *
    277  *                                                                                             *
    278  *    This routine will return the time it takes to construct this object. Usually the time    *
    279  *    to produce is directly related to cost.                                                  *
    280  *                                                                                             *
    281  * INPUT:   none                                                                               *
    282  *                                                                                             *
    283  * OUTPUT:  Returns with the time to produce this object type. The time is expressed in the    *
    284  *          form of game ticks.                                                                *
    285  *                                                                                             *
    286  * WARNINGS:   none                                                                            *
    287  *                                                                                             *
    288  * HISTORY:                                                                                    *
    289  *   07/29/1995 JLB : Created.                                                                 *
    290  *=============================================================================================*/
    291 int TechnoTypeClass::Time_To_Build(HousesType house) const
    292 {
    293 	/*
    294 	**	Base time is the raw cost.
    295 	*/
    296 	int time = Raw_Cost();
    297 
    298 	HouseClass* hptr = HouseClass::As_Pointer(house);
    299 	if (!hptr || !hptr->IsActive) {
    300 		return time;
    301 	}
    302 
    303 	/*
    304 	**	For computer controlled buildings, slow down production on
    305 	**	cheaper buildings.
    306 	*/
    307 	if (PlayerPtr->Difficulty != DIFF_HARD &&
    308 		GameToPlay == GAME_NORMAL &&
    309 		What_Am_I() == RTTI_BUILDINGTYPE &&
    310 		PlayerPtr->Class->House != hptr->Class->House) {
    311 		time = (time + (PlayerPtr->Difficulty == DIFF_EASY ? 4000 : 2000)) / 2;
    312 	}
    313 
    314 	/*
    315 	**	Fudge factor, so that Nod builds a bit faster if the object must be delivered to
    316 	**	an airfield.
    317 	*/
    318 	if (What_Am_I() == RTTI_UNITTYPE && !(Ownable & HOUSEF_GOOD)) {
    319 		time -= (time / 4);
    320 	}
    321 
    322 	/*
    323 	**	Adjust time according to IQ setting of computer controlled house. The
    324 	**	build time will range from double normal time at the slowest to
    325 	**	just normal time at the fastest.
    326 	*/
    327 	if (!hptr->IsHuman && Rule.Diff[hptr->Difficulty].IsBuildSlowdown) {
    328 		time = (int)((time * Rule.MaxIQ * 2.0f) / (hptr->IQ + Rule.MaxIQ));
    329 	}
    330 
    331 	/*
    332 	**	Adjust the time to build based on the power output of the owning house.
    333 	*/
    334 	int power = hptr->Power_Fraction();
    335 	int inv_scale(256);
    336 	if (power == 0x000) {
    337 		inv_scale = 64;
    338 	}
    339 	else if (power < 0x080) {
    340 		inv_scale = 103;
    341 	}
    342 	else if (power < 0x0100) {
    343 		inv_scale = 171;
    344 	}
    345 	time = (time * 256) / inv_scale;
    346 
    347 	return(time);
    348 }
    349 
    350 
    351 /***********************************************************************************************
    352  * TechnoTypeClass::Cost_Of -- Fetches the cost of this object type.                           *
    353  *                                                                                             *
    354  *    This routine will return the cost to produce an object of this type.                     *
    355  *                                                                                             *
    356  * INPUT:   none                                                                               *
    357  *                                                                                             *
    358  * OUTPUT:  Returns with the cost to produce one object of this type.                          *
    359  *                                                                                             *
    360  * WARNINGS:   none                                                                            *
    361  *                                                                                             *
    362  * HISTORY:                                                                                    *
    363  *   07/29/1995 JLB : Created.                                                                 *
    364  *=============================================================================================*/
    365 int TechnoTypeClass::Cost_Of(void) const
    366 {
    367 	return(Cost);
    368 }
    369 
    370 
    371 /***********************************************************************************************
    372  * TechnoTypeClass::Get_Cameo_Data -- Fetches the cameo image for this object type.            *
    373  *                                                                                             *
    374  *    This routine will fetch the cameo (sidebar small image) shape of this object type.       *
    375  *    If there is no cameo data available (typical for non-producable units), then NULL will   *
    376  *    be returned.                                                                             *
    377  *                                                                                             *
    378  * INPUT:   none                                                                               *
    379  *                                                                                             *
    380  * OUTPUT:  Returns with a pointer to the cameo data for this object type if present.          *
    381  *                                                                                             *
    382  * WARNINGS:   none                                                                            *
    383  *                                                                                             *
    384  * HISTORY:                                                                                    *
    385  *   07/29/1995 JLB : Created.                                                                 *
    386  *=============================================================================================*/
    387 void const * TechnoTypeClass::Get_Cameo_Data(void) const
    388 {
    389 	return(CameoData);
    390 }
    391 
    392 
    393 /***********************************************************************************************
    394  * TechnoTypeClass::Repair_Cost -- Fetches the cost to repair one step.                        *
    395  *                                                                                             *
    396  *    This routine will return the cost to repair one step. At the TechnoTypeClass level,      *
    397  *    this merely serves as a placeholder function. The derived classes will provide a         *
    398  *    functional version of this routine.                                                      *
    399  *                                                                                             *
    400  * INPUT:   none                                                                               *
    401  *                                                                                             *
    402  * OUTPUT:  Returns with the cost to repair one step.                                          *
    403  *                                                                                             *
    404  * WARNINGS:   none                                                                            *
    405  *                                                                                             *
    406  * HISTORY:                                                                                    *
    407  *   07/29/1995 JLB : Created.                                                                 *
    408  *=============================================================================================*/
    409 int TechnoTypeClass::Repair_Cost(void) const
    410 {
    411 	return(0);
    412 }
    413 
    414 
    415 /***********************************************************************************************
    416  * TechnoTypeClass::Repair_Step -- Fetches the health to repair one step.                      *
    417  *                                                                                             *
    418  *    This routine merely serves as placeholder virtual function. The various type classes     *
    419  *    will override this routine to return the number of health points to repair in one        *
    420  *    "step".                                                                                  *
    421  *                                                                                             *
    422  * INPUT:   none                                                                               *
    423  *                                                                                             *
    424  * OUTPUT:  Returns with the number of health points to repair in one step.                    *
    425  *                                                                                             *
    426  * WARNINGS:   none                                                                            *
    427  *                                                                                             *
    428  * HISTORY:                                                                                    *
    429  *   07/29/1995 JLB : Created.                                                                 *
    430  *=============================================================================================*/
    431 int TechnoTypeClass::Repair_Step(void) const
    432 {
    433 	return(0);
    434 }
    435 
    436 
    437 #ifdef CHEAT_KEYS
    438 /***********************************************************************************************
    439  * TechnoClass::Debug_Dump -- Displays the base class data to the monochrome screen.           *
    440  *                                                                                             *
    441  *    This routine is used to dump the status of the object class to the monochrome screen.    *
    442  *    This display can be used to track down or prevent bugs.                                  *
    443  *                                                                                             *
    444  * INPUT:   none                                                                               *
    445  *                                                                                             *
    446  * OUTPUT:  none                                                                               *
    447  *                                                                                             *
    448  * WARNINGS:   none                                                                            *
    449  *                                                                                             *
    450  * HISTORY:                                                                                    *
    451  *   06/02/1994 JLB : Created.                                                                 *
    452  *=============================================================================================*/
    453 void TechnoClass::Debug_Dump(MonoClass *mono) const
    454 {
    455 	mono->Set_Cursor(0,0);mono->Printf("(%04X)p=%d,d=%d", House->Power_Fraction(), House->Power, House->Drain);
    456 //	mono->Set_Cursor(0,0);mono->Printf("(%d)", House->Blockage);
    457 	mono->Text_Print("X", 16 + (IsALoaner?2:0), 11);
    458 	mono->Text_Print("X", 16 + (IsLocked?2:0), 9);
    459 
    460 	mono->Text_Print("X", 16 + (IsInRecoilState?2:0), 17);
    461 	mono->Text_Print("X", 16 + (IsTethered?2:0), 8);
    462 	mono->Text_Print("X", 16 + (IsOwnedByPlayer?2:0), 5);
    463 	mono->Text_Print("X", 16 + (IsDiscoveredByPlayer?2:0), 6);
    464 //	mono->Text_Print("X", 16 + (IsALemon?2:0), 9);
    465 	mono->Set_Cursor(28, 7);mono->Printf("%2d", Arm);
    466 	mono->Set_Cursor(34, 1);mono->Printf("%04X", TarCom);
    467 	mono->Set_Cursor(29, 3);mono->Printf("%02X", PrimaryFacing.Current());
    468 
    469 	FlasherClass::Debug_Dump(mono);
    470 	StageClass::Debug_Dump(mono);
    471 	RadioClass::Debug_Dump(mono);
    472 }
    473 #endif
    474 
    475 
    476 /***********************************************************************************************
    477  * TechnoClass::TechnoClass -- Default constructor for techno objects.                         *
    478  *                                                                                             *
    479  *    This default constructor for techno objects is used to reset all internal variables to   *
    480  *    their default state.                                                                     *
    481  *                                                                                             *
    482  * INPUT:   none                                                                               *
    483  *                                                                                             *
    484  * OUTPUT:  none                                                                               *
    485  *                                                                                             *
    486  * WARNINGS:   none                                                                            *
    487  *                                                                                             *
    488  * HISTORY:                                                                                    *
    489  *   12/09/1994 JLB : Created.                                                                 *
    490  *=============================================================================================*/
    491 TechnoClass::TechnoClass(void) :
    492 	TarCom(TARGET_NONE),
    493 	House(0)
    494 {
    495 	ArchiveTarget = TARGET_NONE;
    496 	Arm = 0;
    497 	Ammo = -1;
    498 	LineCount = 0;
    499 	LineFrame = 0;
    500 	LineMaxFrames = 0;
    501 	PurchasePrice = 0;
    502 	IsTickedOff = false;
    503 	Cloak = UNCLOAKED;
    504 	CloakingDevice.Set_Stage(1);
    505 	CloakingDevice.Set_Rate(0);
    506 	IsCloakable = false;
    507 	IsALemon = false;
    508 	IsALoaner = false;
    509 	IsDiscoveredByPlayer = false;
    510 	IsDiscoveredByComputer = false;
    511 	IsInRecoilState = false;
    512 	IsLeader = false;
    513 	IsLocked = false;
    514 	IsOwnedByPlayer = false;
    515 	IsSecondShot = true;
    516 	IsTethered = false;
    517 	SuspendedTarCom = TARGET_NONE;
    518 	PrimaryFacing.Set(DIR_N);
    519 
    520 	// Added for multiplayer changes. ST - 3/6/2019 11:34AM
    521 	IsDiscoveredByPlayerMask = 0;
    522 }
    523 
    524 
    525 
    526 /***********************************************************************************************
    527  * TechnoClass::Revealed -- Handles revealing an object to the house specified.                *
    528  *                                                                                             *
    529  *    When a unit moves out from under the shroud or when it gets delivered into already       *
    530  *    explored terrain, then it must be "revealed". Objects that are revealed may be           *
    531  *    announced to the player. The discovered bit updated accordingly.                         *
    532  *                                                                                             *
    533  * INPUT:   house -- The house that this object is revealed to.                                *
    534  *                                                                                             *
    535  * OUTPUT:  none                                                                               *
    536  *                                                                                             *
    537  * WARNINGS:   none                                                                            *
    538  *                                                                                             *
    539  * HISTORY:                                                                                    *
    540  *   06/02/1994 JLB : Created.                                                                 *
    541  *   12/27/1994 JLB : Discovered trigger event processing.                                     *
    542  *   03/06/2019  ST : Per-player discovery                                                     *
    543  *=============================================================================================*/
    544 bool TechnoClass::Revealed(HouseClass * house)
    545 {
    546 	//if (house == PlayerPtr && IsDiscoveredByPlayer) return(false);
    547 	//if (house != PlayerPtr && IsDiscoveredByComputer) return(false);
    548 	if (house == NULL) {
    549 		return false;
    550 	}
    551 
    552 	if (Is_Discovered_By_Player(house)) {
    553 		return false;
    554 	}
    555 
    556 	if (house->IsHuman == false) {
    557 		if (IsDiscoveredByComputer) {
    558 			return false;
    559 		}
    560 	}
    561 
    562 	if (RadioClass::Revealed(house)) {
    563 
    564 		/*
    565 		**	An enemy object that is discovered will go into hunt mode if
    566 		**	its current mission is to ambush.
    567 		*/
    568 		if (!house->IsHuman && Mission == MISSION_AMBUSH) {
    569 			Assign_Mission(MISSION_HUNT);
    570 		}
    571 
    572 		if (GameToPlay == GAME_NORMAL) {
    573 			if (house == PlayerPtr) {
    574 				if (!IsOwnedByPlayer) {
    575 					
    576 					/*
    577 					**	If there is a trigger event associated with this object, then process
    578 					**	it for discovery purposes.
    579 					*/
    580 					if (Trigger) {
    581 						Trigger->Spring(EVENT_DISCOVERED, this);
    582 					}
    583 
    584 					/*
    585 					**	Alert the enemy house to presence of the friendly side.
    586 					*/
    587 					House->IsDiscovered = true;
    588 				
    589 					if (house->IsHuman) {
    590 						Set_Discovered_By_Player(house);
    591 					} else {
    592 						IsDiscoveredByComputer = true;
    593 					}
    594 
    595 				} else {
    596 					
    597 					if (house->IsHuman) {
    598 						Set_Discovered_By_Player(house);
    599 					} else {
    600 						IsDiscoveredByComputer = true;
    601 					}
    602 					Look();
    603 					  
    604 				}	
    605 			}
    606 		} else {
    607 			
    608 			if (house->IsHuman) {
    609 				Set_Discovered_By_Player(house);
    610 			} else {
    611 				IsDiscoveredByComputer = true;
    612 			}
    613 			
    614 			Look();
    615 		}
    616 
    617 		return(true);
    618 	}
    619 	return(false);
    620 }
    621 
    622 #if (0)	                               // ST - 3/6/2019 11:35AM
    623 /***********************************************************************************************
    624  * TechnoClass::Revealed -- Handles revealing an object to the house specified.                *
    625  *                                                                                             *
    626  *    When a unit moves out from under the shroud or when it gets delivered into already       *
    627  *    explored terrain, then it must be "revealed". Objects that are revealed may be           *
    628  *    announced to the player. The discovered bit updated accordingly.                         *
    629  *                                                                                             *
    630  * INPUT:   house -- The house that this object is revealed to.                                *
    631  *                                                                                             *
    632  * OUTPUT:  none                                                                               *
    633  *                                                                                             *
    634  * WARNINGS:   none                                                                            *
    635  *                                                                                             *
    636  * HISTORY:                                                                                    *
    637  *   06/02/1994 JLB : Created.                                                                 *
    638  *   12/27/1994 JLB : Discovered trigger event processing.                                     *
    639  *=============================================================================================*/
    640 bool TechnoClass::Revealed(HouseClass * house)
    641 {
    642 	if (house == PlayerPtr && IsDiscoveredByPlayer) return(false);
    643 	if (house != PlayerPtr && IsDiscoveredByComputer) return(false);
    644 
    645 	if (RadioClass::Revealed(house)) {
    646 
    647 		/*
    648 		**	An enemy object that is discovered will go into hunt mode if
    649 		**	its current mission is to ambush.
    650 		*/
    651 		if (!house->IsHuman && Mission == MISSION_AMBUSH) {
    652 			Assign_Mission(MISSION_HUNT);
    653 		}
    654 
    655 		if (house == PlayerPtr) {
    656 
    657 			if (!IsOwnedByPlayer) {
    658 
    659 				/*
    660 				**	If there is a trigger event associated with this object, then process
    661 				**	it for discovery purposes.
    662 				*/
    663 				if (Trigger) {
    664 					Trigger->Spring(EVENT_DISCOVERED, this);
    665 				}
    666 
    667 				/*
    668 				**	Alert the enemy house to presence of the friendly side.
    669 				*/
    670 				House->IsDiscovered = true;
    671 
    672 			} else {
    673 
    674 				/*
    675 				**	A newly revealed object will always perform a look operation.
    676 				*/
    677 				if (house == PlayerPtr) IsDiscoveredByPlayer = true;
    678 				if (house != PlayerPtr) IsDiscoveredByComputer = true;
    679 				Look();
    680 			}
    681 		}
    682 
    683 		if (house == PlayerPtr) IsDiscoveredByPlayer = true;
    684 		if (house != PlayerPtr) IsDiscoveredByComputer = true;
    685 
    686 		return(true);
    687 	}
    688 	return(false);
    689 }
    690 #endif
    691 
    692 /***********************************************************************************************
    693  * TechnoClass::Hidden -- Returns the object back into the hidden state.                       *
    694  *                                                                                             *
    695  *    This routine is called for every object that returns to the darkness shroud or when      *
    696  *    it gets destroyed. This also occurs when an object enters another (such as infantry      *
    697  *    entering a transport helicopter).                                                        *
    698  *                                                                                             *
    699  * INPUT:   none                                                                               *
    700  *                                                                                             *
    701  * OUTPUT:  none                                                                               *
    702  *                                                                                             *
    703  * WARNINGS:   none                                                                            *
    704  *                                                                                             *
    705  * HISTORY:                                                                                    *
    706  *   06/02/1994 JLB : Created.                                                                 *
    707  *=============================================================================================*/
    708 void TechnoClass::Hidden(void)
    709 {
    710 	// ST - 3/13/2019 4:56PM
    711 	if (!Is_Discovered_By_Player()) {
    712 		return;
    713 	}
    714 	//if (!IsDiscoveredByPlayer) return;
    715 	if (!House->IsHuman) {
    716 		Clear_Discovered_By_Players();
    717 	}
    718 }
    719 
    720 
    721 /***********************************************************************************************
    722  * TechnoClass::Mark -- Handles marking of techno objects.                                     *
    723  *                                                                                             *
    724  *    On the Techno-level, marking handles transmission of the redraw command to any object    *
    725  *    that it is 'connected' with. This only occurs during loading and unloading.              *
    726  *                                                                                             *
    727  * INPUT:   mark  -- The marking method. This routine just passes this on to base classes.     *
    728  *                                                                                             *
    729  * OUTPUT:  none                                                                               *
    730  *                                                                                             *
    731  * WARNINGS:   none                                                                            *
    732  *                                                                                             *
    733  * HISTORY:                                                                                    *
    734  *   10/17/1994 JLB : Created.                                                                 *
    735  *=============================================================================================*/
    736 bool TechnoClass::Mark(MarkType mark)
    737 {
    738 	if (RadioClass::Mark(mark)) {
    739 		/*
    740 		**	When redrawing an object, if there is another object teathered to this one,
    741 		**	redraw it as well.
    742 		*/
    743 		if (IsTethered) {
    744 			Transmit_Message(RADIO_REDRAW);
    745 		}
    746 		return(true);
    747 	}
    748 	return(false);
    749 }
    750 
    751 
    752 /***********************************************************************************************
    753  * TechnoClass::Receive_Message -- Handles inbound message as appropriate.                     *
    754  *                                                                                             *
    755  *    This routine is used to handle inbound radio messages. It only handles those messages    *
    756  *    that deal with techno objects. Typically, these include loading and unloading.           *
    757  *                                                                                             *
    758  * INPUT:   from     -- Pointer to the originator of the radio message.                        *
    759  *                                                                                             *
    760  *          message  -- The inbound radio message.                                             *
    761  *                                                                                             *
    762  *          param    -- Reference to optional parameter that might be used to transfer         *
    763  *                      more information than is possible with the simple radio message        *
    764  *                      type.                                                                  *
    765  *                                                                                             *
    766  * OUTPUT:  Returns with the radio response.                                                   *
    767  *                                                                                             *
    768  * WARNINGS:   none                                                                            *
    769  *                                                                                             *
    770  * HISTORY:                                                                                    *
    771  *   10/17/1994 JLB : Created.                                                                 *
    772  *   06/17/1995 JLB : Handles tether contact messages.                                         *
    773  *=============================================================================================*/
    774 RadioMessageType TechnoClass::Receive_Message(RadioClass * from, RadioMessageType message, long & param)
    775 {
    776 	switch (message) {
    777 
    778 		/*
    779 		**	Just received instructions to attack the specified target.
    780 		*/
    781 		case RADIO_ATTACK_THIS:
    782 			if (Techno_Type_Class()->Primary != WEAPON_NONE) {
    783 				Assign_Target(param);
    784 				Assign_Mission(MISSION_ATTACK);
    785 				return(RADIO_ROGER);
    786 			}
    787 			break;
    788 
    789 		/*
    790 		**	Establish a tethered connection to the object in radio contact.
    791 		*/
    792 		case RADIO_TETHER:
    793 			if (!IsTethered) {
    794 				IsTethered = true;
    795 				Transmit_Message(RADIO_TETHER, from);
    796 				return(RADIO_ROGER);
    797 			}
    798 			break;
    799 
    800 		/*
    801 		**	Break the tethered connection to the object in radio contact.
    802 		*/
    803 		case RADIO_UNTETHER:
    804 			if (IsTethered) {
    805 				IsTethered = false;
    806 				Transmit_Message(RADIO_UNTETHER, from);
    807 				return(RADIO_ROGER);
    808 			}
    809 			break;
    810 
    811 		/*
    812 		**	A teathered unit has reached it's destination. All is
    813 		**	clear, so radio contact can be broken.
    814 		*/
    815 		case RADIO_UNLOADED:
    816 			Transmit_Message(RADIO_UNTETHER, from);
    817 			return(Transmit_Message(RADIO_OVER_OUT, from));
    818 
    819 		/*
    820 		**	When this message is received, it means that the other object
    821 		**	has already turned its radio off. Turn this radio off as well.
    822 		*/
    823 		case RADIO_OVER_OUT:
    824 			Transmit_Message(RADIO_UNTETHER, from);
    825 			RadioClass::Receive_Message(from, message, param);
    826 			return(RADIO_ROGER);
    827 
    828 		/*
    829 		**	Request to be informed when unloaded. This message is transmitted
    830 		**	by the transport unit to the transported unit as it is about to be
    831 		**	unloaded. It is saying, "All is clear. Drive off now."
    832 		*/
    833 		case RADIO_UNLOAD:
    834 		case RADIO_BACKUP_NOW:
    835 		case RADIO_HOLD_STILL:
    836 			Transmit_Message(RADIO_TETHER, from);
    837 			RadioClass::Receive_Message(from, message, param);
    838 			return(RADIO_ROGER);
    839 
    840 		/*
    841 		**	Handle reloading one ammo point for this unit.
    842 		*/
    843 		case RADIO_RELOAD:
    844 			if (Ammo == Techno_Type_Class()->MaxAmmo) return(RADIO_NEGATIVE);
    845 			Ammo++;
    846 			return(RADIO_ROGER);
    847 
    848 		/*
    849 		**	Handle repair of this unit.
    850 		*/
    851 		case RADIO_REPAIR:
    852 			if (/*param > 0 &&*/ Health_Ratio() < 0x0100) {
    853 				int cost = Techno_Type_Class()->Repair_Cost();
    854 //				int cost = Fixed_To_Cardinal(Techno_Type_Class()->Repair_Cost(), param);
    855 				cost = MAX(cost, 1);
    856 				int step = Techno_Type_Class()->Repair_Step();
    857 //				int step = Fixed_To_Cardinal(Techno_Type_Class()->Repair_Step(), param);
    858 				step = MAX(step, 1);
    859 				if (House->Available_Money() >= cost) {
    860 #ifdef OBSOLETE
    861 					if (Health_Ratio() >= 0x0100) {
    862 						Strength = Class_Of().MaxStrength;
    863 						from->Scatter(true);
    864 						return(RADIO_NEGATIVE);
    865 					}
    866 #endif
    867 					House->Spend_Money(cost);
    868 					Strength += step;
    869 					return(RADIO_ROGER);
    870 				}
    871 			}
    872 			break;
    873 
    874 		default:
    875 			break;
    876 	}
    877 	return(RadioClass::Receive_Message(from, message, param));
    878 }
    879 
    880 
    881 /***********************************************************************************************
    882  * TechnoClass::TechnoClass -- Constructor for techno type objects.                            *
    883  *                                                                                             *
    884  *    This constructor sets the owner of this object and its strength. Any object not owned    *
    885  *    by the player is marked as a loaner. This is a special flag that indicates off-map       *
    886  *    movement is allowed. The flag is cleared when the object finally enters the map.         *
    887  *                                                                                             *
    888  * INPUT:   house    -- The house (owner) of this object.                                      *
    889  *                                                                                             *
    890  *          strength -- The strength to assign to this object.                                 *
    891  *                                                                                             *
    892  * OUTPUT:  none                                                                               *
    893  *                                                                                             *
    894  * WARNINGS:   none                                                                            *
    895  *                                                                                             *
    896  * HISTORY:                                                                                    *
    897  *   10/17/1994 JLB : Created.                                                                 *
    898  *=============================================================================================*/
    899 TechnoClass::TechnoClass(HousesType house) :
    900 	House(HouseClass::As_Pointer(house)),
    901 	TarCom(TARGET_NONE)
    902 {
    903 	ArchiveTarget = TARGET_NONE;
    904 	Arm = 0;
    905 	Ammo = -1;
    906 	LineCount = 0;
    907 	LineFrame = 0;
    908 	LineMaxFrames = 0;
    909 	PurchasePrice = 0;
    910 	IsTickedOff = false;
    911 	Cloak = UNCLOAKED;
    912 	CloakingDevice.Set_Stage(1);
    913 	CloakingDevice.Set_Rate(0);
    914 	IsCloakable = false;
    915 	IsALemon = false;
    916 	IsALoaner = false;
    917 	IsDiscoveredByComputer = false;
    918 	//IsOwnedByPlayer = (house == PlayerPtr->Class->House);  // ST - 4/24/2019 10:41AM		
    919 	IsDiscoveredByPlayer = false;
    920 	IsInRecoilState = false;
    921 	IsLeader = false;
    922 	IsLocked = false;
    923 	IsSecondShot = false;
    924 	IsTethered = false;
    925 
    926 	SuspendedTarCom = TARGET_NONE;
    927 
    928 	PrimaryFacing.Set(DIR_N);
    929 
    930 	// Added for multiplayer changes. ST - 4/24/2019 10:40AM
    931 	IsDiscoveredByPlayerMask = 0;
    932 	if (GameToPlay == GAME_NORMAL) {
    933 		IsOwnedByPlayer = (house == PlayerPtr->Class->House);
    934 	} else {
    935 		IsOwnedByPlayer = House->IsHuman;
    936 	}
    937 
    938 	/*
    939 	**	There is a chance that a vehicle will be a "lemon".
    940 	*/
    941 	if (Random_Pick(0, 255) < (int)House->Class->Lemon) {
    942 		IsALemon = true;
    943 	}
    944 }
    945 
    946 
    947 /***********************************************************************************************
    948  * TechnoClass::Per_Cell_Process -- Handles once-per-cell operations for techno type objects.  *
    949  *                                                                                             *
    950  *    This routine handles marking a game object as not a loaner. It is set only if the unit   *
    951  *    is not player owned and is on the regular map. This is necessary so that enemy objects   *
    952  *    can exist off-map but as soon as they move onto the map, are flagged so that can never   *
    953  *    leave it again.                                                                          *
    954  *                                                                                             *
    955  * INPUT:   none                                                                               *
    956  *                                                                                             *
    957  * OUTPUT:  none                                                                               *
    958  *                                                                                             *
    959  * WARNINGS:   none                                                                            *
    960  *                                                                                             *
    961  * HISTORY:                                                                                    *
    962  *   10/17/1994 JLB : Created.                                                                 *
    963  *   10/26/94   JLB : Handles scanner units.                                                   *
    964  *   12/27/1994 JLB : Checks for an processes any trigger in cell.                             *
    965  *=============================================================================================*/
    966 void TechnoClass::Per_Cell_Process(bool )
    967 {
    968 	CELL cell = Coord_Cell(Center_Coord());
    969 
    970 	/*
    971 	**	When enemy units enter the proper map area from off map, they are
    972 	**	flagged so that they won't travel back off the map again.
    973 	*/
    974 	if (Map.In_Radar(cell)) {
    975 
    976 		if (What_Am_I() == RTTI_UNIT) {
    977 			UnitClass * u = (UnitClass *)this;
    978 
    979 			if (*u != UNIT_HOVER && *u != UNIT_GUNBOAT) {
    980 				IsLocked = true;
    981 			}
    982 		} else {
    983 			IsLocked = true;
    984 		}
    985 	}
    986 
    987 	/*
    988 	**	If this object somehow moves into mapped terrain, but is not yet
    989 	**	discovered, then flag it to be discovered.
    990 	*/
    991 	// ST - 3/13/2019 4:56PM
    992 	if (!Is_Discovered_By_Player() && Map[cell].Is_Visible(House)) {
    993 	//if (!IsDiscoveredByPlayer && Map[cell].IsVisible) {
    994 		//Revealed(PlayerPtr);
    995 		Revealed(House);		// ST - 8/7/2019 11:20AM
    996 	}
    997 }
    998 
    999 
   1000 /***********************************************************************************************
   1001  * TechnoClass::Draw_It -- Draws the health bar (if necessary).                                *
   1002  *                                                                                             *
   1003  *    This routine will draw the common elements for techno type objects. This element is      *
   1004  *    the health bar. The main game object has already been rendered by the time this          *
   1005  *    routine is called.                                                                       *
   1006  *                                                                                             *
   1007  * INPUT:   x,y   -- The coordinate of the center of the unit.                                 *
   1008  *                                                                                             *
   1009  * OUTPUT:  none                                                                               *
   1010  *                                                                                             *
   1011  * WARNINGS:   none                                                                            *
   1012  *                                                                                             *
   1013  * HISTORY:                                                                                    *
   1014  *   10/17/1994 JLB : Created.                                                                 *
   1015  *   10/26/94   JLB : Knows about radar scanned cells.                                         *
   1016  *   12/13/1994 JLB : Clips health bar against map edge.                                       *
   1017  *   01/23/1995 JLB : Dynamic selected object rectangle.                                       *
   1018  *=============================================================================================*/
   1019 void TechnoClass::Draw_It(int x, int y, WindowNumberType window)
   1020 {
   1021 	Clear_Redraw_Flag();
   1022 
   1023 	const bool show_health_bar =
   1024 		(Strength > 0) && !Is_Cloaked(PlayerPtr) &&
   1025 		(Is_Selected_By_Player() ||
   1026 		((Special.HealthBarDisplayMode == SpecialClass::HB_DAMAGED) && (Strength < Techno_Type_Class()->MaxStrength)) ||
   1027 		(Special.HealthBarDisplayMode == SpecialClass::HB_ALWAYS));
   1028 
   1029 	/*
   1030 	**	Fetch the dimensions of the object. These dimensions will be used to draw
   1031 	**	the selection box and the health bar.
   1032 	*/
   1033 	int width, height;
   1034 	Class_Of().Dimensions(width, height);
   1035 	int lx = width/2;
   1036 	int ly = height/2;
   1037 	int dx = width/5;
   1038 	int dy = height/5;
   1039 	int fudge = show_health_bar ? 4 : 0;
   1040 
   1041 	/*
   1042 	** Draw lines
   1043 	*/
   1044 	if (LineFrame < LineMaxFrames) {
   1045 		// Only draw the last line for virtual window
   1046 		int start_line = (window == WINDOW_VIRTUAL) ? max(0, LineCount - 1) : 0;
   1047 		for (int i = start_line; i < LineCount; i++) {
   1048 			CC_Draw_Line(Lines[i][0], Lines[i][1], Lines[i][2], Lines[i][3], (unsigned char)Lines[i][4], LineFrame, window);
   1049 		}
   1050 		if (window == WINDOW_VIRTUAL) {
   1051 			LineFrame++;
   1052 		}
   1053 	}
   1054 
   1055 	if (Is_Selected_By_Player() || show_health_bar) {
   1056 		GraphicViewPortClass draw_window(	LogicPage->Get_Graphic_Buffer(),
   1057 														(WindowList[window][WINDOWX] << 3) + LogicPage->Get_XPos(),
   1058 														WindowList[window][WINDOWY] + LogicPage->Get_YPos(),
   1059 														WindowList[window][WINDOWWIDTH] << 3,
   1060 														WindowList[window][WINDOWHEIGHT]);
   1061 
   1062 
   1063 		/*
   1064 		**	The infantry select box should be a bit higher than normal.
   1065 		*/
   1066 		if (What_Am_I() == RTTI_INFANTRY) {
   1067 			y -= 6;
   1068 		}
   1069 
   1070 		if (What_Am_I() == RTTI_BUILDING && ((BuildingTypeClass const &)Class_Of()).Type == STRUCT_BARRACKS) {
   1071 			y -= 5;
   1072 		}
   1073 
   1074 		if (show_health_bar) {
   1075 			unsigned 	ratio = Health_Ratio();
   1076 			int			pwidth;		// Pixel width of bar interior.
   1077 			int			color;		// The color to give the interior of the bargraph.
   1078 
   1079 			int xx = x-width/2;
   1080 			int yy = y-(height/2);
   1081 
   1082 			/*
   1083 			**	Draw the outline of the bargraph.
   1084 			*/
   1085 			draw_window.Remap(xx+1, yy+1, width-1, 3-1, Map.FadingShade);
   1086 			draw_window.Draw_Rect(xx, yy, xx+width-1, yy+3, BLACK);
   1087 
   1088 			/*
   1089 			**	Determine the width of the interior strength
   1090 			**	graph.
   1091 			*/
   1092 			pwidth = Fixed_To_Cardinal(width-2, ratio);
   1093 
   1094 			pwidth = Bound(pwidth, 1, width-2);
   1095 
   1096 			color = LTGREEN;
   1097 			if (ratio < 0x7F) {
   1098 				color = YELLOW;
   1099 			}
   1100 			if (ratio < 0x3F) {
   1101 				color = RED;
   1102 			}
   1103 			draw_window.Fill_Rect(xx+1, yy+1, xx+pwidth, yy+(3-1), color);
   1104 		}
   1105 
   1106 		/*
   1107 		**	Draw the selected object graphic.
   1108 		*/
   1109 		if (Is_Selected_By_Player()) {
   1110 			// Upper left corner.
   1111 			draw_window.Draw_Line(x-lx, fudge+y-ly, x-lx+dx, fudge+y-ly, WHITE);
   1112 			draw_window.Draw_Line(x-lx, fudge+y-ly, x-lx, fudge+y-ly+dy, WHITE);
   1113 
   1114 			// Upper right corner.
   1115 			draw_window.Draw_Line(x+lx, fudge+y-ly, x+lx-dx, fudge+y-ly, WHITE);
   1116 			draw_window.Draw_Line(x+lx, fudge+y-ly, x+lx, fudge+y-ly+dy, WHITE);
   1117 
   1118 			// Lower right corner.
   1119 			draw_window.Draw_Line(x+lx, y+ly, x+lx-dx, y+ly, WHITE);
   1120 			draw_window.Draw_Line(x+lx, y+ly, x+lx, y+ly-dy, WHITE);
   1121 
   1122 			// Lower left corner.
   1123 			draw_window.Draw_Line(x-lx, y+ly, x-lx+dx, y+ly, WHITE);
   1124 			draw_window.Draw_Line(x-lx, y+ly, x-lx, y+ly-dy, WHITE);
   1125 		}
   1126 	}
   1127 
   1128 	// MBL 04.21.2020 
   1129 	bool selected = Is_Selected_By_Player() || Special.ResourceBarDisplayMode == SpecialClass::RB_ALWAYS;
   1130 	// if ((window == WINDOW_VIRTUAL) || (Is_Selected_By_Player() && House->Is_Ally(PlayerPtr))) 
   1131 	if ((window == WINDOW_VIRTUAL) || (selected && House->Is_Ally(PlayerPtr))) 
   1132 	{
   1133 		Draw_Pips((x-lx)+5, y+ly-3, window);
   1134 	}
   1135 }
   1136 
   1137 
   1138 /***********************************************************************************************
   1139  * TechnoClass::Unlimbo -- Performs unlimbo process for all techno type objects.               *
   1140  *                                                                                             *
   1141  *    This routine handles the common operation between techno objects when they are           *
   1142  *    unlimboed. This includes revealing the map.                                              *
   1143  *                                                                                             *
   1144  * INPUT:   coord    -- The coordinate to unlimbo object at.                                   *
   1145  *                                                                                             *
   1146  *          dir (optional) -- initial facing direction for this object                         *
   1147  *                                                                                             *
   1148  * OUTPUT:  bool; Was the unlimbo successful?                                                  *
   1149  *                                                                                             *
   1150  * WARNINGS:   none                                                                            *
   1151  *                                                                                             *
   1152  * HISTORY:                                                                                    *
   1153  *   11/14/1994 JLB : Created.                                                                 *
   1154  *=============================================================================================*/
   1155 bool TechnoClass::Unlimbo(COORDINATE coord, DirType dir)
   1156 {
   1157 	if (RadioClass::Unlimbo(coord, dir)) {
   1158 		PrimaryFacing = dir;
   1159 		Enter_Idle_Mode(true);
   1160 		Commence();
   1161 
   1162 		IsLocked = Map.In_Radar(Coord_Cell(coord));
   1163 		return(true);
   1164 	}
   1165 	return(false);
   1166 }
   1167 
   1168 
   1169 /***********************************************************************************************
   1170  * TechnoClass::In_Range -- Determines if specified target is within weapon range.             *
   1171  *                                                                                             *
   1172  *    This routine is used to compare the distance to the specified target with the range      *
   1173  *    of the weapon. If the target is outside of weapon range, then false is returned.         *
   1174  *                                                                                             *
   1175  * INPUT:   target   -- The target to check if it is within weapon range.                      *
   1176  *                                                                                             *
   1177  *          which    -- Which weapon to use in determining range. 0=primary, 1=secondary.      *
   1178  *                                                                                             *
   1179  * OUTPUT:  bool; Is the specified target within weapon range?                                 *
   1180  *                                                                                             *
   1181  * WARNINGS:   none                                                                            *
   1182  *                                                                                             *
   1183  * HISTORY:                                                                                    *
   1184  *   11/14/1994 JLB : Created.                                                                 *
   1185  *=============================================================================================*/
   1186 bool TechnoClass::In_Range(TARGET target, int which, bool reciprocal_check) const
   1187 {
   1188 	if (IsLocked && Target_Legal(target)) {
   1189 		int range = Weapon_Range(which);
   1190 		BuildingClass const * building = As_Building(target);
   1191 		if (building) {
   1192 			range += ((building->Class->Width() + building->Class->Height()) * (ICON_LEPTON_W / 4));
   1193 		}
   1194 		FireDataType data = Fire_Data(which);
   1195 		if (MAX(0, ::Distance(data.Center, As_Coord(target)) - data.Distance) <= range) {
   1196 			return(true);
   1197 		}
   1198 		
   1199 		/*
   1200 		** There's a longstanding bug where enemy units aligned north/south may have different perceptions of whether they
   1201 		** are in range of each other due to the turrets offset on the unit. This check will say that if either considers the 
   1202 		** the other to be in range, then they both do. ST - 3/18/2020 10:44AM
   1203 		*/
   1204 		if (reciprocal_check && !building) {
   1205 			ObjectClass *target_object = As_Object(target);
   1206 			if (target_object) {
   1207 				RTTIType my_type = What_Am_I();
   1208 				if (target_object->What_Am_I() == my_type) {
   1209 					if (range == target_object->Weapon_Range(which)) {
   1210 						TechnoClass *tech = As_Techno(target);
   1211 						if (tech->In_Range(As_Target(), which, false)) {
   1212 							return true;
   1213 						}
   1214 					}
   1215 				}
   1216 			}
   1217 		}
   1218 	}
   1219 	return(false);
   1220 }
   1221 
   1222 
   1223 /***********************************************************************************************
   1224  * TechnoClass::In_Range -- Determines if specified target is within weapon range.             *
   1225  *                                                                                             *
   1226  *    This routine will determine if the pointer to the target object passed into this         *
   1227  *    routine is within weapon range.                                                          *
   1228  *                                                                                             *
   1229  * INPUT:   target   -- Pointer to the target object to check if within weapon range.          *
   1230  *                                                                                             *
   1231  *          which    -- Which weapon to use in determining range. 0=primary, 1=secondary.      *
   1232  *                                                                                             *
   1233  * OUTPUT:  bool; Is the target within weapon range?                                           *
   1234  *                                                                                             *
   1235  * WARNINGS:   none                                                                            *
   1236  *                                                                                             *
   1237  * HISTORY:                                                                                    *
   1238  *   11/14/1994 JLB : Created.                                                                 *
   1239  *=============================================================================================*/
   1240 bool TechnoClass::In_Range(ObjectClass const * target, int which, bool reciprocal_check) const
   1241 {
   1242 	if (IsLocked && target) {
   1243 		int range = Weapon_Range(which);
   1244 		if (target->What_Am_I() == RTTI_BUILDING) {
   1245 			BuildingClass const * building = (BuildingClass const *)target;
   1246 			range += ((building->Class->Width() + building->Class->Height()) * (ICON_LEPTON_W / 4));
   1247 		}
   1248 		FireDataType data = Fire_Data(which);
   1249 		if (MAX(0, ::Distance(data.Center, target->Center_Coord()) - data.Distance) <= range) {
   1250 			return(true);
   1251 		}
   1252 		
   1253 		/*
   1254 		** There's a longstanding bug where enemy units aligned north/south may have different perceptions of whether they
   1255 		** are in range of each other due to the turrets offset on the unit. This check will say that if either considers the 
   1256 		** the other to be in range, then they both do. ST - 3/18/2020 10:44AM
   1257 		*/
   1258 		if (reciprocal_check) {
   1259 			RTTIType my_type = What_Am_I();
   1260 			if (my_type != RTTI_BUILDING) {
   1261 				if (target->What_Am_I() == my_type) {
   1262 					if (range == target->Weapon_Range(which)) {
   1263 						if (((TechnoClass*)target)->In_Range(this, which, false)) {
   1264 							return true;
   1265 						}
   1266 					}
   1267 				}
   1268 			}
   1269 		}
   1270 	}
   1271 	return(false);
   1272 }
   1273 
   1274 
   1275 /***********************************************************************************************
   1276  * TechnoClass::In_Range -- Determines if the specified coordinate is within range.            *
   1277  *                                                                                             *
   1278  *    Use this routine to determine if the specified coordinate is within weapon range.        *
   1279  *                                                                                             *
   1280  * INPUT:   coord    -- The coordinate to examine against the object to determine range.       *
   1281  *                                                                                             *
   1282  *          which    -- The weapon to consider when determining range. 0=primary, 1=secondary. *
   1283  *                                                                                             *
   1284  * OUTPUT:  bool; Is the weapon within range?                                                  *
   1285  *                                                                                             *
   1286  * WARNINGS:   none                                                                            *
   1287  *                                                                                             *
   1288  * HISTORY:                                                                                    *
   1289  *   03/16/1995 JLB : Created.                                                                 *
   1290  *=============================================================================================*/
   1291 bool TechnoClass::In_Range(COORDINATE coord, int which) const
   1292 {
   1293 	return(IsLocked && ::Distance(Fire_Coord(which), coord) <= Weapon_Range(which));
   1294 }
   1295 
   1296 
   1297 /***********************************************************************************************
   1298  * TechnoClass::Evaluate_Object -- Determines score value of specified object.                 *
   1299  *                                                                                             *
   1300  *    This routine is used to determing the score value (value as a potential target) of the   *
   1301  *    object specified. This routine will check the specified object for all the various       *
   1302  *    legality checks that threat scanning requires. This is the main workhorse routine for    *
   1303  *    target searching.                                                                        *
   1304  *                                                                                             *
   1305  * INPUT:   method   -- The threat method requested. This is a combined bitflag value that     *
   1306  *                      not only specified the kind of targets to consider, but how far away   *
   1307  *                      they are allowed to be.                                                *
   1308  *                                                                                             *
   1309  *          mask     -- This is an RTTI mask to use for quickly eliminating object types.      *
   1310  *                      The mask is created outside of this routine because this routine is    *
   1311  *                      usually called from within a loop and this value is constant in that   *
   1312  *                      context.                                                               *
   1313  *                                                                                             *
   1314  *          range    -- The range at which potential target objects are rejected.              *
   1315  *                      0  = must be within weapon range.                                      *
   1316  *                      >0 = must be within this lepton distance.                              *
   1317  *                      <0 = range doesn't matter.                                             *
   1318  *                                                                                             *
   1319  *          object   -- Pointer to the object itself.                                          *
   1320  *                                                                                             *
   1321  *          value    -- Reference to the value variable that this routine will fill in. The    *
   1322  *                      higher the value the more likely this object will be selected as best. *
   1323  *                                                                                             *
   1324  * OUTPUT:  Did the target pass all legality checks? If this value is returned true, then the  *
   1325  *          value parameter will be filled in correctly.                                       *
   1326  *                                                                                             *
   1327  * WARNINGS:   This routine is time consuming. Don't call unless necessary.                    *
   1328  *                                                                                             *
   1329  * HISTORY:                                                                                    *
   1330  *   06/30/1995 JLB : Created.                                                                 *
   1331  *   07/14/1995 JLB : Forces SAM site to not fire on landed aircraft.                          *
   1332  *=============================================================================================*/
   1333 bool TechnoClass::Evaluate_Object(ThreatType method, int mask, int range, TechnoClass const * object, int & value) const
   1334 {
   1335 	/*
   1336 	**	An object in limbo can never be a valid target.
   1337 	*/
   1338 	if (object->IsInLimbo) return(false);
   1339 
   1340 	/*
   1341 	**	Friendly units are never considered a good target. Bail if this
   1342 	**	object is a friend.
   1343 	*/
   1344 	if (House->Is_Ally(object)) return(false);
   1345 
   1346 	/*
   1347 	**	If the object is farther away than allowed, bail.
   1348 	*/
   1349 	int dist = Distance(object);
   1350 	if (range > 0 && dist > range) return(false);
   1351 	if (range == 0 && !In_Range(object, 0) && !In_Range(object, 1)) return(false);
   1352 
   1353 	/*
   1354 	**	If the object is not visible, then bail. Human controled units
   1355 	**	are always considered to be visible.
   1356 	*/
   1357 	if (!object->IsOwnedByPlayer && !object->IsDiscoveredByPlayer && GameToPlay == GAME_NORMAL && object->What_Am_I() != RTTI_AIRCRAFT) {
   1358 		return(false);
   1359 	}
   1360 
   1361 	/*
   1362 	**	Quickly eliminate all unit types that are not allowed according to the mask
   1363 	**	value.
   1364 	*/
   1365 	RTTIType otype = object->What_Am_I();
   1366 	if (!((1 << otype) & mask)) return(false);		// Mask failure.
   1367 
   1368 	/*
   1369 	**	If the object is cloaked, then it isn't a legal target.
   1370 	*/
   1371 	if (object->Is_Cloaked(this)) return(false);
   1372 
   1373 	/*
   1374 	**	Determine if the target is theoretically allowed to be a target. If
   1375 	**	not, then bail.
   1376 	*/
   1377 	TechnoTypeClass const * tclass = object->Techno_Type_Class();
   1378 	if (!tclass->IsLegalTarget) return(false);		// Legality failure.
   1379 
   1380 	/*
   1381 	**	Never consider agent Delphi a valid target.
   1382 	*/
   1383 	if (otype == RTTI_INFANTRY && ((InfantryTypeClass const *)tclass)->Type == INFANTRY_DELPHI) {
   1384 		return(false);
   1385 	}
   1386 
   1387 	/*
   1388 	**	Special case so that SAM site doesn't fire on aircraft that are landed.
   1389 	*/
   1390 	if (otype == RTTI_AIRCRAFT && What_Am_I() == RTTI_BUILDING && *((BuildingClass *)this) == STRUCT_SAM) {
   1391 		if (((AircraftClass *)object)->Altitude == 0) return(false);
   1392 	}
   1393 
   1394 	/*
   1395 	**	If only allowed to attack civilians, then eliminate all other types.
   1396 	*/
   1397 	if ((method & THREAT_CIVILIANS) && object->Owner() != HOUSE_NEUTRAL) {
   1398 		return(false);
   1399 	}
   1400 
   1401 	/*
   1402 	**	If the scan is limited to capturable buildings only, then bail if the examined
   1403 	**	object isn't a capturable building.
   1404 	*/
   1405 	if ((method & THREAT_CAPTURE) && (otype != RTTI_BUILDING || !object->Can_Capture())) {
   1406 		return(false);
   1407 	}
   1408 
   1409 	/*
   1410 	**	If not allowed to attack boats, then eliminate them from consideration.
   1411 	*/
   1412 	if (!(method & THREAT_BOATS) &&
   1413 			otype == RTTI_UNIT &&
   1414 			(((UnitTypeClass const *)tclass)->Speed == SPEED_HOVER || ((UnitTypeClass const *)tclass)->Speed == SPEED_FLOAT)) {
   1415 		return(false);
   1416 	}
   1417 
   1418 	/*
   1419 	**	SPECIAL CASE: Friendly units won't automatically fire on buildings
   1420 	**	if the building is not aggressive.
   1421 	*/
   1422 	if (House->IsHuman && otype == RTTI_BUILDING && tclass->Primary == WEAPON_NONE) return(false);
   1423 
   1424 	/*
   1425 	**	If the search is restricted to Tiberium processing objects, then
   1426 	**	perform the special qualification check now.
   1427 	*/
   1428 	if (method & THREAT_TIBERIUM) {
   1429 		switch (otype) {
   1430 			case RTTI_UNIT:
   1431 				if (!((UnitTypeClass const *)tclass)->IsToHarvest) return(false);
   1432 				break;
   1433 
   1434 			case RTTI_BUILDING:
   1435 				if (!((BuildingTypeClass const *)tclass)->Capacity) return(false);
   1436 				break;
   1437 
   1438 			default:
   1439 				return(false);
   1440 		}
   1441 	}
   1442 
   1443 	/*
   1444 	**	If this target value is better than the previously recorded best
   1445 	**	target value then record this target for possible return as the
   1446 	**	best target available.
   1447 	*/
   1448 	int rawval = object->Value();
   1449 	value = rawval + object->Kills;
   1450 
   1451 #ifdef USE_RA_AI
   1452 	/*
   1453 	**	If the candidate object is owned by the designated enemy of this house, then
   1454 	**	give it a higher value. This will tend to gravitate attacks toward the main
   1455 	**	antagonist of this house.
   1456 	*/
   1457 	if (House->Enemy != HOUSE_NONE && House->Enemy == object->House->Class->House) {
   1458 		value += 500;
   1459 		value *= 3;
   1460 	}
   1461 #endif
   1462 
   1463 #ifdef ADVANCED
   1464 	/*
   1465 	**	Lessen threat as a factor of distance.
   1466 	*/
   1467 	if (rawval) {
   1468 
   1469 		value = (value * 32000) / (((dist/ICON_LEPTON_W)*(dist/ICON_LEPTON_W))+1);
   1470 
   1471 		//value = Fixed_To_Cardinal(value, Cardinal_To_Fixed(MAP_CELL_W*2, (MAP_CELL_W*2) - (dist/ICON_LEPTON_W)));
   1472 		//value = MAX(value, 2);
   1473 
   1474 		if (value < MAP_CELL_W*2) value = dist/ICON_LEPTON_W;
   1475 		value = MAX(value, 1);
   1476 		return(true);
   1477 	}
   1478 	value = 0;
   1479 	return(false);
   1480 
   1481 #else
   1482 
   1483 	/*
   1484 	**	Lessen threat as a factor of distance.
   1485 	*/
   1486 	if (range == -1 && value) {
   1487 		/*
   1488 		** Code from RA so that the value isn't always 2 on a mapwide scan with the range set to -1. ST - 3/2/2020 5:02PM
   1489 		*/
   1490 		value = (value * 32000) / ((dist/ICON_LEPTON_W)+1);
   1491 	} else {
   1492 		/*
   1493 		** Original TD code
   1494 		*/
   1495 		int modifier = dist;
   1496 		int crange = range / ICON_LEPTON_W;
   1497 		if (crange) modifier /= crange;
   1498 		if (modifier) value /= modifier;
   1499 	}
   1500 	if (rawval) {
   1501 		value = MAX(value, 2);
   1502 	}
   1503 	return(true);
   1504 #endif
   1505 }
   1506 
   1507 
   1508 /***********************************************************************************************
   1509  * TechnoClass::Evaluate_Cell -- Determine the value and object of specified cell.             *
   1510  *                                                                                             *
   1511  *    This routine will examine the specified cell and return with the potential target        *
   1512  *    object it contains and the value of it. Use this routine when searching for threats.     *
   1513  *                                                                                             *
   1514  * INPUT:   method   -- The scan method to use for target searching.                           *
   1515  *                                                                                             *
   1516  *          mask     -- Prebuilt mask of object RTTI types acceptable for scanning.            *
   1517  *                                                                                             *
   1518  *          range    -- Scan range limit to use for elimination purposes. This ensures that    *
   1519  *                      objects in the "corner" of a square scan get properly discarded.       *
   1520  *                                                                                             *
   1521  *          object   -- Pointer to object pointer to be filled in with the object at this      *
   1522  *                      cell as a valid target.                                                *
   1523  *                                                                                             *
   1524  *          value    -- Reference to the value of the object in this cell. It will be set      *
   1525  *                      according to the object's value.                                       *
   1526  *                                                                                             *
   1527  * OUTPUT:  Was a valid potential target found in this cell?                                   *
   1528  *                                                                                             *
   1529  * WARNINGS:   none                                                                            *
   1530  *                                                                                             *
   1531  * HISTORY:                                                                                    *
   1532  *   06/19/1995 JLB : Created.                                                                 *
   1533  *=============================================================================================*/
   1534 bool TechnoClass::Evaluate_Cell(ThreatType method, int mask, CELL cell, int range, TechnoClass const * * object, int & value) const
   1535 {
   1536 	*object = NULL;
   1537 	value = 0;
   1538 
   1539 	/*
   1540 	**	If the cell is not on the legal map, then always ignore it.
   1541 	*/
   1542 	if (cell & 0xF000) return(false);
   1543 	if (!Map.In_Radar(cell)) return(false);
   1544 
   1545 	/*
   1546 	**	Fetch the techno object from the cell. If there is no
   1547 	**	techno object there, then bail.
   1548 	*/
   1549 	CellClass * cellptr = &Map[cell];
   1550 	TechnoClass const * tentative = (TechnoClass const *)cellptr->Cell_Occupier();
   1551 	while (tentative) {
   1552 		if (tentative->Is_Techno() && !House->Is_Ally(tentative)) break;
   1553 		tentative = (TechnoClass const *)tentative->Next;
   1554 	}
   1555 
   1556 	if (!tentative) return(false);
   1557 //	if (!tentative->Is_Techno()) return(false);
   1558 	*object = tentative;
   1559 
   1560 	return(Evaluate_Object(method, mask, range, tentative, value));
   1561 }
   1562 
   1563 
   1564 /***********************************************************************************************
   1565  * TechnoClass::Greatest_Threat -- Determines best target given search criteria.               *
   1566  *                                                                                             *
   1567  *    This routine will scan game objects looking for the best target. It is used by the       *
   1568  *    general target searching processes. The type of target scan to perform is controlled     *
   1569  *    by the method control parameter.                                                         *
   1570  *                                                                                             *
   1571  * INPUT:   method   -- The method control parameter is used to control the type of target     *
   1572  *                      scan performed. It consists of a series of bit flags (see ThreatType)  *
   1573  *                      that are combined to form the target scan desired.                     *
   1574  *                                                                                             *
   1575  * OUTPUT:  Returns the target value of a suitable target. If no target was found then the     *
   1576  *          value TARGET_NONE is returned.                                                     *
   1577  *                                                                                             *
   1578  * WARNINGS:   none                                                                            *
   1579  *                                                                                             *
   1580  * HISTORY:                                                                                    *
   1581  *   11/14/1994 JLB : Created.                                                                 *
   1582  *   06/20/1995 JLB : Greatly optimized scan method.                                           *
   1583  *=============================================================================================*/
   1584 TARGET TechnoClass::Greatest_Threat(ThreatType method) const
   1585 {
   1586 	ObjectClass const * bestobject = NULL;
   1587 	int bestval = -1;
   1588 
   1589 	/*
   1590 	**	Build a quick elimination mask. If the RTTI of the object doesn't
   1591 	**	qualify with this mask, then we KNOW that it shouldn't be considered.
   1592 	*/
   1593 	int mask = 0;
   1594 	if (method & THREAT_CIVILIANS) mask |= ((1 << RTTI_BUILDING) | (1 << RTTI_INFANTRY) | (1 << RTTI_UNIT));
   1595 	if (method & THREAT_AIR) mask |= (1 << RTTI_AIRCRAFT);
   1596 	if (method & THREAT_CAPTURE) mask |= (1 << RTTI_BUILDING);
   1597 	if (method & THREAT_BUILDINGS) mask |= (1 << RTTI_BUILDING);
   1598 	if (method & THREAT_INFANTRY) mask |= (1 << RTTI_INFANTRY);
   1599 	if (method & THREAT_VEHICLES) mask |= (1 << RTTI_UNIT);
   1600 
   1601 	/*
   1602 	** Let's pick from the available contenders on a map-wide scan. ST - 3/2/2020 3:49PM
   1603 	*/
   1604 	static const int _max_best_objects = 128;
   1605 	ObjectClass const *best_objects [_max_best_objects];
   1606 	int best_object_count = 0;
   1607 
   1608 	/*
   1609 	**	Limit area target scans use a method where the actual map cells are
   1610 	**	examined for occupants. The occupant is then examined in turn. The
   1611 	**	best target within the area is returned as a target.
   1612 	*/
   1613 	if (method & (THREAT_AREA|THREAT_RANGE)) {
   1614 		int range = Threat_Range((method & THREAT_RANGE) ? 0 : 1);
   1615 
   1616 //		int range = MAX(Weapon_Range(0), Weapon_Range(1));
   1617 //		if (!(method & THREAT_RANGE)) range *= 2;
   1618 //		range = Bound(range, 0x0100, 0x1400);			// Limit maximum scan distance.
   1619 		int crange = range / ICON_LEPTON_W;
   1620 		if (range == 0) {
   1621 			crange = MAX(Weapon_Range(0), Weapon_Range(1)) / ICON_LEPTON_W;
   1622 			crange++;
   1623 		}
   1624 		CELL cell = Coord_Cell(Fire_Coord(0));
   1625 //		CELL cell = Coord_Cell(Center_Coord());
   1626 
   1627 		/*
   1628 		**	If aircraft are a legal target, then scan through all of them at this time.
   1629 		**	Scanning by cell is not possible for aircraft since they are not recorded
   1630 		**	at the cell level.
   1631 		*/
   1632 		if (method & THREAT_AIR) {
   1633 			for (int index = 0; index < Aircraft.Count(); index++) {
   1634 				TechnoClass * object = Aircraft.Ptr(index);
   1635 
   1636 				int value = 0;
   1637 				if (Evaluate_Object(method, mask, range, object, value)) {
   1638 					if (value > bestval) {
   1639 						bestobject = object;
   1640 						bestval = value;
   1641 					}
   1642 				}
   1643 			}
   1644 		}
   1645 
   1646 		/*
   1647 		**	When scanning the ground, always consider landed aircraft as a valid
   1648 		**	potential target. This is only true if vehicles are considered a
   1649 		**	valid target. A landed aircraft is considered a vehicle.
   1650 		*/
   1651 		if (method & THREAT_VEHICLES) {
   1652 			mask |= (1 << RTTI_AIRCRAFT);
   1653 		}
   1654 
   1655 		/*
   1656 		**	Radiate outward from the object's location, looking for the best
   1657 		**	target.
   1658 		*/
   1659 		TechnoClass const * object;
   1660 		int value;
   1661 		for (int radius = 1; radius < crange; radius++) {
   1662 
   1663 			/*
   1664 			**	Scan the top and bottom rows of the "box".
   1665 			*/
   1666 			for (int x = -radius; x <= radius; x++) {
   1667 				CELL newcell;
   1668 
   1669 				if ((Cell_X(cell) + x) < Map.MapCellX) continue;
   1670 				if ((Cell_X(cell) + x) >= (Map.MapCellX+Map.MapCellWidth)) continue;
   1671 
   1672 				if ((Cell_Y(cell) - radius) >= Map.MapCellY) {
   1673 					newcell = XY_Cell(Cell_X(cell) + x, Cell_Y(cell)-radius);
   1674 					if (Evaluate_Cell(method, mask, newcell, range, &object, value)) {
   1675 						if (bestval < value) {
   1676 							bestobject = object;
   1677 						}
   1678 					}
   1679 				}
   1680 
   1681 				if ((Cell_Y(cell) + radius) < (Map.MapCellY+Map.MapCellHeight)) {
   1682 					newcell = XY_Cell(Cell_X(cell)+x, Cell_Y(cell)+radius);
   1683 					if (Evaluate_Cell(method, mask, newcell, range, &object, value)) {
   1684 						if (bestval < value) {
   1685 							bestobject = object;
   1686 						}
   1687 					}
   1688 				}
   1689 			}
   1690 
   1691 			/*
   1692 			**	Scan the left and right columns of the "box".
   1693 			*/
   1694 			for (int y = -(radius-1); y < radius; y++) {
   1695 				CELL newcell;
   1696 
   1697 				if ((Cell_Y(cell) + y) < Map.MapCellY) continue;
   1698 				if ((Cell_Y(cell) + y) >= (Map.MapCellY+Map.MapCellHeight)) continue;
   1699 
   1700 				if ((Cell_X(cell) - radius) >= Map.MapCellX) {
   1701 					newcell = XY_Cell(Cell_X(cell)-radius, Cell_Y(cell)+y);
   1702 					if (Evaluate_Cell(method, mask, newcell, range, &object, value)) {
   1703 						if (bestval < value) {
   1704 							bestobject = object;
   1705 						}
   1706 					}
   1707 				}
   1708 
   1709 				if ((Cell_X(cell) + radius) < (Map.MapCellX+Map.MapCellWidth)) {
   1710 					newcell = XY_Cell(Cell_X(cell)+radius, Cell_Y(cell)+y);
   1711 					if (Evaluate_Cell(method, mask, newcell, range, &object, value)) {
   1712 						if (bestval < value) {
   1713 							bestobject = object;
   1714 						}
   1715 					}
   1716 				}
   1717 			}
   1718 
   1719 			/*
   1720 			**	Bail early if a target has already been found and the range is at
   1721 			**	one of the breaking points (i.e., normal range or range * 2).
   1722 			*/
   1723 			if (bestobject) {
   1724 				if (radius == crange/4) {
   1725 					return(bestobject->As_Target());
   1726 				}
   1727 				if (radius == crange/2) {
   1728 					return(bestobject->As_Target());
   1729 				}
   1730 			}
   1731 		}
   1732 
   1733 	} else {
   1734 
   1735 		/*
   1736 		**	A full map scan was requested. First scan through aircraft. The top map layer
   1737 		**	is NOT scanned since that layer will probably contain more bullets and animations
   1738 		**	than aircraft.
   1739 		*/
   1740 		int index;
   1741 		for (index = 0; index < Aircraft.Count(); index++) {
   1742 			TechnoClass * object = Aircraft.Ptr(index);
   1743 
   1744 			int value = 0;
   1745 			if (Evaluate_Object(method, mask, -1, object, value)) {
   1746 				if (value > bestval) {
   1747 					bestobject = object;
   1748 					bestval = value;
   1749 					best_object_count = 0;
   1750 					best_objects[best_object_count++] = object;
   1751 				} else {
   1752 					if (value == bestval) {
   1753 						if (best_object_count < _max_best_objects) {
   1754 							best_objects[best_object_count++] = object;
   1755 						}
   1756 					}
   1757 				}
   1758 			}
   1759 		}
   1760 
   1761 		/*
   1762 		**	Now scan through the entire ground layer. This is painful, but what other
   1763 		**	choice is there?
   1764 		*/
   1765 		for (index = 0; index < Map.Layer[LAYER_GROUND].Count(); index++) {
   1766 			ObjectClass const * object = Map.Layer[LAYER_GROUND][index];
   1767 
   1768 			int value = 0;
   1769 			if (object->Is_Techno() && Evaluate_Object(method, mask, -1, (TechnoClass const *)object, value)) {
   1770 				if (value > bestval) {
   1771 					bestobject = object;
   1772 					bestval = value;
   1773 					best_object_count = 0;
   1774 					best_objects[best_object_count++] = object;
   1775 				} else {
   1776 					if (value == bestval) {
   1777 						if (best_object_count < _max_best_objects) {
   1778 							best_objects[best_object_count++] = object;
   1779 						}
   1780 					}
   1781 				}
   1782 			}
   1783 		}
   1784 	}
   1785 
   1786 	/*
   1787 	**	If a good target object was found, then return with the target value
   1788 	**	of it.
   1789 	*/
   1790 	if (bestobject) {
   1791 		
   1792 		/*
   1793 		** If there's only one, return that. Otherwise pick from the equal contenders
   1794 		*/
   1795 		if (best_object_count > 1) {
   1796 
   1797 			/*
   1798 			** Pick one at random from our contenders
   1799 			*/
   1800 			int index = Random_Pick(0, best_object_count - 1);
   1801 			bestobject = As_Object(best_objects[index]->As_Target());
   1802 		}
   1803 
   1804 		return(bestobject->As_Target());
   1805 	}
   1806 	return(TARGET_NONE);
   1807 }
   1808 
   1809 
   1810 /***********************************************************************************************
   1811  * TechnoClass::Owner -- Who is the owner of this object?                                      *
   1812  *                                                                                             *
   1813  *    Use this routine to examine this object and return who the owner is.                     *
   1814  *                                                                                             *
   1815  * INPUT:   none                                                                               *
   1816  *                                                                                             *
   1817  * OUTPUT:  Returns with the house number of the owner of this object.                         *
   1818  *                                                                                             *
   1819  * WARNINGS:   none                                                                            *
   1820  *                                                                                             *
   1821  * HISTORY:                                                                                    *
   1822  *   12/09/1994 JLB : Created.                                                                 *
   1823  *=============================================================================================*/
   1824 HousesType TechnoClass::Owner(void) const
   1825 {
   1826 	return(House->Class->House);
   1827 }
   1828 
   1829 
   1830 /***********************************************************************************************
   1831  * TechnoClass::Clicked_As_Target -- Sets the flash count for this techno object.              *
   1832  *                                                                                             *
   1833  *    Use this routine to set the flash count for the object. This flash count is the number   *
   1834  *    of times the object will "flash". Typically it is called as a result of the player       *
   1835  *    clicking on this object in order to make it the target of a move or attack.              *
   1836  *                                                                                             *
   1837  * INPUT:   count -- The number of times the object should flash.                              *
   1838  *                                                                                             *
   1839  * OUTPUT:  none                                                                               *
   1840  *                                                                                             *
   1841  * WARNINGS:   none                                                                            *
   1842  *                                                                                             *
   1843  * HISTORY:                                                                                    *
   1844  *   12/09/1994 JLB : Created.                                                                 *
   1845  *=============================================================================================*/
   1846 void TechnoClass::Clicked_As_Target(HousesType house, int count) // 2019/09/20 JAS - Added record of who clicked on the object
   1847 {
   1848 	FlashCount = count;
   1849 
   1850 	// 2019/09/20 JAS - Flashing info needs to exist per player
   1851 	if (house < HOUSE_COUNT)
   1852 	{
   1853 		FlashCountPerPlayer[house] = count;
   1854 	}
   1855 	else
   1856 	{
   1857 		//receiving HOUSE_COUNT means do it for every player
   1858 		for (int i = 0; i < HOUSE_COUNT; ++i)
   1859 		{
   1860 			FlashCountPerPlayer[i] = count;
   1861 		}
   1862 	}
   1863 }
   1864 
   1865 
   1866 /***********************************************************************************************
   1867  * TechnoClass::AI -- Handles AI processing for techno object.                                 *
   1868  *                                                                                             *
   1869  *    This routine handles AI processing for techno objects. Typically, this merely dispatches *
   1870  *    to the appropriate AI routines for the base classes.                                     *
   1871  *                                                                                             *
   1872  * INPUT:   none                                                                               *
   1873  *                                                                                             *
   1874  * OUTPUT:  none                                                                               *
   1875  *                                                                                             *
   1876  * WARNINGS:   Make sure that this routine is only called ONCE per game tick.                  *
   1877  *                                                                                             *
   1878  * HISTORY:                                                                                    *
   1879  *   12/09/1994 JLB : Created.                                                                 *
   1880  *=============================================================================================*/
   1881 void TechnoClass::AI(void)
   1882 {
   1883 	CargoClass::AI();
   1884 	RadioClass::AI();
   1885 	DoorClass::AI();
   1886 
   1887 	/*
   1888 	** Handle decision to re-cloak here. Process the cloaking/decloaking operation.
   1889 	*/
   1890 	if (IsCloakable) {
   1891 
   1892 		/*
   1893 		**	If this object is uncloaked, but it can be cloaked and it thinks that it
   1894 		**	is a good time do so, then begin cloaking.
   1895 		*/
   1896 		if (Cloak == UNCLOAKED) {
   1897 			// Changed for multiplayer so we can visually see the different players in the original renderer. ST - 3/13/2019 5:40PM
   1898 			if (Is_Owned_By_Player()) Mark(MARK_CHANGE);
   1899 			//if (IsOwnedByPlayer) Mark(MARK_CHANGE);
   1900 			CloakingDevice.Graphic_Logic();
   1901 			if (!Arm && CloakingDevice.Fetch_Stage()) {
   1902 				if (Health_Ratio() > 0x0040) {
   1903 					Do_Cloak();
   1904 				} else {
   1905 					if (Random_Pick(0, 25) == 1) {
   1906 						Do_Cloak();
   1907 					}
   1908 				}
   1909 			}
   1910 		} else {
   1911 
   1912 			VisualType pre = Visual_Character(true);
   1913 			CloakingDevice.Graphic_Logic();
   1914 			switch (Cloak) {
   1915 
   1916 				/*
   1917 				**	Handle the uncloaking process. Always mark to redraw
   1918 				**	the object and when cloaking is complete, stabilize into
   1919 				**	the normal uncloaked state.
   1920 				*/
   1921 				case UNCLOAKING:
   1922 					Mark(MARK_CHANGE);
   1923 					if (Visual_Character(true) == VISUAL_NORMAL) {
   1924 						CloakingDevice.Set_Rate(UNCLOAK_VIS_TIME);
   1925 						CloakingDevice.Set_Stage(0);	// re-start the stage counter
   1926 						Cloak = UNCLOAKED;
   1927 					}
   1928 					break;
   1929 
   1930 				/*
   1931 				**	Handle the cloaking process. Always mark to redraw the object
   1932 				**	and when the cloaking process is complete, stabilize into the
   1933 				**	normal cloaked state.
   1934 				*/
   1935 				case CLOAKING:
   1936 					Mark(MARK_CHANGE);
   1937 					switch (Visual_Character(true)) {
   1938 
   1939 						/*
   1940 						**	If badly damaged, then it can never fully cloak.
   1941 						*/
   1942 						case VISUAL_DARKEN:
   1943 							if (Health_Ratio() < 0x0040 && Random_Pick(1, 3) == 1) {
   1944 								Cloak = UNCLOAKING;
   1945 							}
   1946 							break;
   1947 
   1948 #ifdef NEVER
   1949 						case VISUAL_SHADOWY:
   1950 							if (pre != Visual_Character(true)) {
   1951 								Detach_All(false);
   1952 							}
   1953 							break;
   1954 #endif
   1955 
   1956 						case VISUAL_HIDDEN:
   1957 							Cloak = CLOAKED;
   1958 							CloakingDevice.Set_Rate(0);
   1959 
   1960 							/*
   1961 							**	Special check to ensure that if the unit is carring a captured
   1962 							**	flag, it will never fully cloak.
   1963 							*/
   1964 							if (What_Am_I() == RTTI_UNIT && ((UnitClass *)this)->Flagged != HOUSE_NONE) {
   1965 								Do_Shimmer();
   1966 							} else {
   1967 								Detach_All(false);
   1968 							}
   1969 
   1970 							/*
   1971 							**	A computer controlled unit will try to scatter if possible so
   1972 							**	that it will be much harder to locate.
   1973 							*/
   1974 							if (What_Am_I() == RTTI_UNIT && !House->IsHuman) {
   1975 								Scatter(0, true);
   1976 							}
   1977 							break;
   1978 					}
   1979 					break;
   1980 
   1981 				/*
   1982 				**	A cloaked object will always be redrawn if it is owned by the
   1983 				**	player. This ensures that the shimmering effect will animate.
   1984 				*/
   1985 				case CLOAKED:
   1986 					// Changed for multiplayer so we can visually see the different players in the original renderer. ST - 3/13/2019 5:40PM
   1987 					if (Is_Owned_By_Player()) {
   1988 					//if (IsOwnedByPlayer) {
   1989 						Mark(MARK_CHANGE);
   1990 					}
   1991 					break;
   1992 			}
   1993 		}
   1994 	}
   1995 
   1996 	/*
   1997 	**	Arming delay always counts down to zero.
   1998 	*/
   1999 	if (Arm) Arm--;
   2000 
   2001 	/*
   2002 	**	Handle line delay logic.
   2003 	*/
   2004 	if (LineMaxFrames > 0) {
   2005 		Map.Flag_To_Redraw(true);
   2006 		if (LineFrame >= LineMaxFrames) {
   2007 			LineCount = 0;
   2008 			LineFrame = 0;
   2009 			LineMaxFrames = 0;
   2010 		}
   2011 	}
   2012 
   2013 	/*
   2014 	**	Update the animation timer system. If the animation stage
   2015 	**	changes, then flag the object to be redrawn as well as  determine
   2016 	**	if the current animation process needs to change.
   2017 	*/
   2018 	if (What_Am_I() != RTTI_BUILDING) {
   2019 		if (StageClass::Graphic_Logic() || Time_To_Redraw()) {
   2020 			Mark(MARK_CHANGE);
   2021 		}
   2022 	}
   2023 
   2024 	/*
   2025 	**	If the object is flashing and a change of flash state has occured, then mark the
   2026 	**	object to be redrawn.
   2027 	*/
   2028 	if (FlasherClass::Process()) {
   2029 		Mark(MARK_CHANGE);
   2030 	}
   2031 }
   2032 
   2033 
   2034 /***********************************************************************************************
   2035  * TechnoClass::Select -- Selects object and checks to see if can be selected.                 *
   2036  *                                                                                             *
   2037  *    This function checks to see if this techno object can be selected. If it can, then it    *
   2038  *    is selected.                                                                             *
   2039  *                                                                                             *
   2040  * INPUT:   none                                                                               *
   2041  *                                                                                             *
   2042  * OUTPUT:  none                                                                               *
   2043  *                                                                                             *
   2044  * WARNINGS:   none                                                                            *
   2045  *                                                                                             *
   2046  * HISTORY:                                                                                    *
   2047  *   12/11/1994 JLB : Created.                                                                 *
   2048  *=============================================================================================*/
   2049 bool TechnoClass::Select(bool allow_mixed)
   2050 {
   2051 	// ST - 3/13/2019 4:56PM
   2052 	if (!Is_Discovered_By_Player() && !Is_Owned_By_Player() && !Debug_Unshroud) {
   2053 	//if (!IsDiscoveredByPlayer && !IsOwnedByPlayer && !Debug_Unshroud) {
   2054 		return(false);
   2055 	}
   2056 
   2057 	if (RadioClass::Select(allow_mixed)) {
   2058 
   2059 		/*
   2060 		**	Speak a confirmation of selection.
   2061 		*/
   2062 		// ST - 3/13/2019 4:59PM
   2063 		if (Is_Owned_By_Player() && AllowVoice) {
   2064 		//if (IsOwnedByPlayer && AllowVoice) {
   2065 			Response_Select();
   2066 		}
   2067 		return(true);
   2068 	}
   2069 	return(false);
   2070 }
   2071 
   2072 
   2073 /***********************************************************************************************
   2074  * TechnoClass::Can_Fire -- Determines if this techno object can fire.                         *
   2075  *                                                                                             *
   2076  *    This performs a simple check to make sure that this techno object can fire. At this      *
   2077  *    level, the only thing checked for is the rearming delay.                                 *
   2078  *                                                                                             *
   2079  * INPUT:   none                                                                               *
   2080  *                                                                                             *
   2081  * OUTPUT:  Returns with the fire legality control code.                                       *
   2082  *                                                                                             *
   2083  * WARNINGS:   none                                                                            *
   2084  *                                                                                             *
   2085  * HISTORY:                                                                                    *
   2086  *   12/23/1994 JLB : Created.                                                                 *
   2087  *=============================================================================================*/
   2088 FireErrorType TechnoClass::Can_Fire(TARGET target, int which) const
   2089 {
   2090 	/*
   2091 	**	Don't allow firing if the target is illegal.
   2092 	*/
   2093 	if (!Target_Legal(target)) {
   2094 		return(FIRE_ILLEGAL);
   2095 	}
   2096 
   2097 	ObjectClass * object = As_Object(target);
   2098 
   2099 	/*
   2100 	**	If the object is completely cloaked, then you can't fire on it.
   2101 	*/
   2102 //Mono_Printf("Units[0]=%p.\n", Units.Raw_Ptr(0));
   2103 //Mono_Printf("Infantry[0]=%p.\n", Infantry.Raw_Ptr(0));
   2104 //Mono_Printf("Buildings[0]=%p.\n", Buildings.Raw_Ptr(0));
   2105 //Mono_Printf("Aircraft[0]=%p.\n", Aircraft.Raw_Ptr(0));
   2106 //Mono_Printf("object=%p, Strength=%d, IsActive=%d, IsInLimbo=%d.\n", object, (long)object->Strength, object->IsActive, object->IsInLimbo);Get_Key();
   2107 	if (object && /*(object->IsActive || GameToPlay != GAME_NORMAL) &&*/ object->Is_Techno() && ((TechnoClass *)object)->Is_Cloaked(this)) {
   2108 		return(FIRE_CANT);
   2109 	}
   2110 
   2111 	/*
   2112 	**	If there is no weapon, then firing is not allowed.
   2113 	*/
   2114 	WeaponType weap = (which == 0) ? Techno_Type_Class()->Primary : Techno_Type_Class()->Secondary;
   2115 	if (weap == WEAPON_NONE) {
   2116 		return(FIRE_CANT);
   2117 	}
   2118 
   2119 	/*
   2120 	**	Can only fire anti-aircraft weapons against aircraft unless the aircraft is
   2121 	**	sitting on the ground.
   2122 	*/
   2123 	if (object && object->What_Am_I() == RTTI_AIRCRAFT &&
   2124 			!BulletTypeClass::As_Reference(Weapons[weap].Fires).IsAntiAircraft &&
   2125 			((AircraftClass *)object)->Altitude > 0) {
   2126 
   2127 		return(FIRE_CANT);
   2128 	}
   2129 
   2130 	/*
   2131 	**	Don't allow firing if still rearming.
   2132 	*/
   2133 	if (Arm) return(FIRE_REARM);
   2134 
   2135 	/*
   2136 	**	The target must be within range in order to allow firing.
   2137 	*/
   2138 	if (!In_Range(target, which)) {
   2139 		return(FIRE_RANGE);
   2140 	}
   2141 
   2142 	/*
   2143 	**	If there is no ammo left, then it can't fire.
   2144 	*/
   2145 	if (!Ammo) {
   2146 		return(FIRE_AMMO);
   2147 	}
   2148 
   2149 	/*
   2150 	**	If cloaked, then firing is disabled.
   2151 	*/
   2152 	if (Cloak != UNCLOAKED) {
   2153 		return(FIRE_CLOAKED);
   2154 	}
   2155 
   2156 	return(FIRE_OK);
   2157 }
   2158 
   2159 
   2160 /***********************************************************************************************
   2161  * TechnoClass::Stun -- Prepares the object for removal from the game.                         *
   2162  *                                                                                             *
   2163  *    This routine handles cleaning up this techno object from the game system so that when    *
   2164  *    it is subsequently removed, it doesn't leave any loose ends.                             *
   2165  *                                                                                             *
   2166  * INPUT:   none                                                                               *
   2167  *                                                                                             *
   2168  * OUTPUT:  none                                                                               *
   2169  *                                                                                             *
   2170  * WARNINGS:   none                                                                            *
   2171  *                                                                                             *
   2172  * HISTORY:                                                                                    *
   2173  *   12/23/1994 JLB : Created.                                                                 *
   2174  *=============================================================================================*/
   2175 void TechnoClass::Stun(void)
   2176 {
   2177 	Assign_Target(TARGET_NONE);
   2178 	Assign_Destination(TARGET_NONE);
   2179 	Transmit_Message(RADIO_OVER_OUT);
   2180 	Detach_All();
   2181 	//Unselect();
   2182 	//When an object is stunned it needs to be deselected from all players, not just the current PlayerPtr.
   2183 	// - 8/18/2019 JAS 
   2184 	Unselect_All_Players();
   2185 }
   2186 
   2187 
   2188 /***********************************************************************************************
   2189  * TechnoClass::Assign_Target -- Assigns the targeting computer with specified target.         *
   2190  *                                                                                             *
   2191  *    Use this routine to set the targeting computer for this object. It checks to make sure   *
   2192  *    that targeting of itself is prohibited.                                                  *
   2193  *                                                                                             *
   2194  * INPUT:   target   -- The target for this object to attack.                                  *
   2195  *                                                                                             *
   2196  * OUTPUT:  none                                                                               *
   2197  *                                                                                             *
   2198  * WARNINGS:   none                                                                            *
   2199  *                                                                                             *
   2200  * HISTORY:                                                                                    *
   2201  *   12/23/1994 JLB : Created.                                                                 *
   2202  *=============================================================================================*/
   2203 void TechnoClass::Assign_Target(TARGET target)
   2204 {
   2205 	if (target == TarCom) return;
   2206 
   2207 	if (!Target_Legal(target)) {
   2208 		target = TARGET_NONE;
   2209 	} else {
   2210 
   2211 		/*
   2212 		**	Prevent targeting of self.
   2213 		*/
   2214 		if (target == As_Target()) {
   2215 			target = ::As_Target(Coord_Cell(Coord));
   2216 		} else {
   2217 
   2218 			/*
   2219 			**	Make sure that the target is not already dead.
   2220 			*/
   2221 			ObjectClass * object = As_Object(target);
   2222 			if (object && (object->IsActive == false || object->Strength == 0)) {
   2223 				target = TARGET_NONE;
   2224 			}
   2225 		}
   2226 	}
   2227 
   2228 	/*
   2229 	**	Set the unit's targeting computer.
   2230 	*/
   2231 	TarCom = target;
   2232 }
   2233 
   2234 
   2235 /***********************************************************************************************
   2236  * TechnoClass::Rearm_Delay -- Calculates the delay before firing can occur.                   *
   2237  *                                                                                             *
   2238  *    This function calculates the delay between shots. It determines this from the standard   *
   2239  *    rate of fire (ROF) of the base class and modifies it according to game speed and         *
   2240  *    whether this is the first or second shot. All single shot attackers consider their       *
   2241  *    shots to be "second" since the second shot is the one handled normally. The first shot   *
   2242  *    usually gets assigned a much shorter delay time before the next shot can fire.           *
   2243  *                                                                                             *
   2244  * INPUT:   second   -- bool; Is this the second of a two shot salvo?                          *
   2245  *                                                                                             *
   2246  * OUTPUT:  Returns with the number of game frames to delay before the next shot may fire.     *
   2247  *                                                                                             *
   2248  * WARNINGS:   none                                                                            *
   2249  *                                                                                             *
   2250  * HISTORY:                                                                                    *
   2251  *   12/26/1994 JLB : Created.                                                                 *
   2252  *=============================================================================================*/
   2253 int TechnoClass::Rearm_Delay(bool second) const
   2254 {
   2255 	if (second) {
   2256 		return(((int)Weapons[Techno_Type_Class()->Primary].ROF * House->ROFBias) + 3);
   2257 	}
   2258 	return(9);
   2259 }
   2260 
   2261 
   2262 /***********************************************************************************************
   2263  * TechnoClass::Fire_At -- Fires projectile at target specified.                               *
   2264  *                                                                                             *
   2265  *    This is the main projectile firing code. Buildings, units, and infantry route fire       *
   2266  *    requests through this function.                                                          *
   2267  *                                                                                             *
   2268  * INPUT:   target   -- The target that the projectile is to be fired at.                      *
   2269  *                                                                                             *
   2270  *          which    -- Which weapon to fire.                                                  *
   2271  *                                                                                             *
   2272  * OUTPUT:  Returns with a pointer to the projectile object that was fired. If no projectile   *
   2273  *          could be created or there was some other illegality detected, the return value     *
   2274  *          will be NULL.                                                                      *
   2275  *                                                                                             *
   2276  * WARNINGS:   none                                                                            *
   2277  *                                                                                             *
   2278  * HISTORY:                                                                                    *
   2279  *   12/26/1994 JLB : Created.                                                                 *
   2280  *   07/03/1995 JLB : Moving platforms fire inaccurate projectiles.                            *
   2281  *=============================================================================================*/
   2282 BulletClass * TechnoClass::Fire_At(TARGET target, int which)
   2283 {
   2284 	BulletClass		*bullet;				// Projectile.
   2285 	DirType			dir;					// The facing to impart upon the projectile.
   2286 	COORDINATE				target_coord;		// Coordinate of the target.
   2287 	COORDINATE				fire_coord;			// Coordinate of firing position.
   2288 	TechnoTypeClass const & tclass = *Techno_Type_Class();
   2289 	ObjectClass		*object;
   2290 	if ((which == 0 && tclass.Primary == WEAPON_NONE) || (which != 0 && tclass.Secondary == WEAPON_NONE)) {
   2291 		return(NULL);
   2292 	}
   2293 	WeaponTypeClass const *weapon = (which == 0) ? &Weapons[tclass.Primary] : &Weapons[tclass.Secondary];
   2294 	BulletTypeClass const &btype = BulletTypeClass::As_Reference(weapon->Fires);
   2295 
   2296 	/*
   2297 	**	Perform a quick legality check to see if firing can occur.
   2298 	*/
   2299 	if (Debug_Map || weapon->Fires == BULLET_NONE || !Target_Legal(target)) {
   2300 		return(NULL);
   2301 	}
   2302 
   2303 	/*
   2304 	**	Fetch the target coordinate for the target specified.
   2305 	*/
   2306 	object = As_Object(target);
   2307 	if (object) {
   2308 		target_coord = object->Target_Coord();
   2309 	} else {
   2310 		target_coord = As_Coord(target);
   2311 	}
   2312 
   2313 	/*
   2314 	**	Get the location where the projectile should appear.
   2315 	*/
   2316 	fire_coord = Fire_Coord(which);
   2317 
   2318 	/*
   2319 	**	If the projectile is a homing type (such as a missile), then it will
   2320 	**	launch in the direction the turret is facing, NOT necessarily the same
   2321 	**	direction as the target.
   2322 	*/
   2323 	if (btype.IsHoming || btype.IsDropping) {
   2324 		dir = Fire_Direction();
   2325 		if (btype.IsDropping) {
   2326 			fire_coord = Center_Coord();
   2327 		}
   2328 	} else {
   2329 		dir = ::Direction(fire_coord, target_coord);
   2330 	}
   2331 
   2332 	/*
   2333 	**	Create the projectile. Then process any special operations that
   2334 	**	need to be performed according to the style of projectile
   2335 	**	created.
   2336 	*/
   2337 	int firepower = (weapon->Attack > 0) ? ((int)weapon->Attack * House->FirepowerBias) : 0;
   2338  	bullet = new BulletClass(weapon->Fires);
   2339 	if (bullet) {
   2340 		bullet->Assign_Target(target);
   2341 		bullet->Payback = this;
   2342 		bullet->Strength = (short)firepower;
   2343 
   2344 		/*
   2345 		**	If this is firing from a moving platform, then the projectile is inaccurate.
   2346 		*/
   2347 		if (Special.IsDefenderAdvantage && What_Am_I() != RTTI_BUILDING && ((FootClass const *)this)->IsDriving) {
   2348 			bullet->IsInaccurate = true;
   2349 		}
   2350 
   2351 		if (bullet->Unlimbo(fire_coord, dir)) {
   2352 //Mono_Printf("Units[0]=%p.\n", Units.Raw_Ptr(0));
   2353 //Mono_Printf("Infantry[0]=%p.\n", Infantry.Raw_Ptr(0));
   2354 //Mono_Printf("Buildings[0]=%p.\n", Buildings.Raw_Ptr(0));
   2355 //Mono_Printf("Aircraft[0]=%p.\n", Aircraft.Raw_Ptr(0));
   2356 //Mono_Printf("object=%p, Strength=%d, IsActive=%d, IsInLimbo=%d.\n", object, (long)object->Strength, object->IsActive, object->IsInLimbo);Get_Key();
   2357 			bullet->Payback = this;
   2358 			bullet->Strength = (short)firepower;
   2359 		} else {
   2360 			delete bullet;
   2361 		}
   2362 		if (!bullet->Class->IsFueled) {
   2363 			IsInRecoilState = true;
   2364 		}
   2365 		Arm = Rearm_Delay(IsSecondShot);
   2366 		if (tclass.IsTwoShooter) {
   2367 			IsSecondShot = (IsSecondShot == false);
   2368 		}
   2369 
   2370 		/*
   2371 		**	Perform any animation effect for this weapon.
   2372 		*/
   2373 		AnimType a = weapon->Anim;
   2374 		switch (a) {
   2375 			case ANIM_GUN_N:
   2376 			case ANIM_CHEM_N:
   2377 			case ANIM_FLAME_N:
   2378 				a = (AnimType)(a + Dir_Facing(Fire_Direction()));
   2379 				break;
   2380 		}
   2381 
   2382 		/*
   2383 		**	If there is a special firing animation, then create and attach it
   2384 		**	now.
   2385 		*/
   2386 		if (a != ANIM_NONE) {
   2387 			AnimClass * anim = new AnimClass(a, Fire_Coord(which));
   2388 			if (anim) {
   2389 				anim->Attach_To(this);
   2390 			}
   2391 		}
   2392 
   2393 		/*
   2394 		**	Reduce ammunition for this object.
   2395 		*/
   2396 		if (Ammo > 0) {
   2397 			Ammo--;
   2398 		}
   2399 
   2400 		/*
   2401 		**	Firing will in all likelihood, require the unit to be redraw. Flag it to be
   2402 		**	redrawn here.
   2403 		*/
   2404 		Mark(MARK_CHANGE);
   2405 
   2406 		/*
   2407 		**	If a projectile was fired from a unit that is hidden in the darkness,
   2408 		**	reveal that unit and a little area around it.
   2409 		** For multiplayer games, only reveal the unit if the target is the
   2410 		** local player.
   2411 		*/
   2412 #if (0)
   2413 		if ((!IsOwnedByPlayer && !IsDiscoveredByPlayer) || !Map[Coord_Cell(Center_Coord())].IsMapped) {
   2414 			if (GameToPlay == GAME_NORMAL) {
   2415 				Map.Sight_From(Coord_Cell(Center_Coord()), 1, false);
   2416 			} else {
   2417 				ObjectClass *obj = As_Object(target);
   2418 				if (obj) {
   2419 					HousesType tgt_owner = obj->Owner();
   2420 
   2421 					if (PlayerPtr->Class->House == tgt_owner) {
   2422 						Map.Sight_From(Coord_Cell(Center_Coord()), 1, false);
   2423 					}
   2424 				}
   2425 			}
   2426 		}
   2427 #else
   2428 		/*
   2429 		** Now need to reveal for any player (only humans in normal node) that is the target. ST - 3/13/2019 5:43PM
   2430 		*/
   2431 
   2432 		ObjectClass *obj = As_Object(target);
   2433 		if (obj) {
   2434 			HousesType tgt_owner = obj->Owner();
   2435 
   2436 			HouseClass *player = HouseClass::As_Pointer(tgt_owner);
   2437 			if (player != nullptr && (player->IsHuman || GameToPlay != GAME_NORMAL)) {
   2438 				if ((!Is_Owned_By_Player(player) && !Is_Discovered_By_Player(player)) || !Map[Coord_Cell(Center_Coord())].Is_Mapped(House)) {
   2439 					Map.Sight_From(player, Coord_Cell(Center_Coord()), 1, false);
   2440 				}
   2441 			}
   2442 		}
   2443 
   2444 #endif
   2445 	}
   2446 
   2447 	if (weapon->Fires == BULLET_LASER) {
   2448 		int x, y, x1, y1;
   2449 		COORDINATE source, dest;
   2450 		source = Fire_Coord(which);
   2451 		dest = As_Coord(target);
   2452 		if (SpecialDialog == SDLG_NONE) {
   2453 			Map.Coord_To_Pixel(source, x, y);
   2454 			Map.Coord_To_Pixel(dest, x1, y1);
   2455 			x += Map.TacPixelX;
   2456 			x1 += Map.TacPixelX;
   2457 			y += Map.TacPixelY;
   2458 			y1 += Map.TacPixelY;
   2459 			Lines[0][0] = x + 1;
   2460 			Lines[0][1] = y;
   2461 			Lines[0][2] = x1;
   2462 			Lines[0][3] = y1;
   2463 			Lines[0][4] = 0x7D;
   2464 			Lines[1][0] = x - 1;
   2465 			Lines[1][1] = y;
   2466 			Lines[1][2] = x1;
   2467 			Lines[1][3] = y1;
   2468 			Lines[1][4] = 0x7D;
   2469 			Lines[2][0] = x;
   2470 			Lines[2][1] = y;
   2471 			Lines[2][2] = x1;
   2472 			Lines[2][3] = y1;
   2473 			Lines[2][4] = 0x7F;
   2474 			LineCount = 3;
   2475 			LineFrame = 0;
   2476 			LineMaxFrames = 5;
   2477 			Map.Flag_To_Redraw(true);
   2478 		}
   2479 		new SmudgeClass(Random_Pick(SMUDGE_SCORCH1, SMUDGE_SCORCH6), As_Coord(target));
   2480 	}
   2481 
   2482 	return(bullet);
   2483 }
   2484 
   2485 
   2486 /***********************************************************************************************
   2487  * TechnoClass::Player_Assign_Mission -- Assigns a mission as result of player input.          *
   2488  *                                                                                             *
   2489  *    This routine is called when the mission for an object needs to change as a result of     *
   2490  *    player input. The basic operation would be to queue the event and let the action         *
   2491  *    occur at the frame dictated by the queing system. However, if a voice response is        *
   2492  *    indicated, then perform it at this time. This will give a greater illusion of            *
   2493  *    immediate response.                                                                      *
   2494  *                                                                                             *
   2495  * INPUT:   order       -- The mission order to assign to this object.                         *
   2496  *                                                                                             *
   2497  *          target      -- The target of this object. This will be used for combat and attack. *
   2498  *                                                                                             *
   2499  *          destination -- The movement destination for this object.                           *
   2500  *                                                                                             *
   2501  * OUTPUT:  none                                                                               *
   2502  *                                                                                             *
   2503  * WARNINGS:   none                                                                            *
   2504  *                                                                                             *
   2505  * HISTORY:                                                                                    *
   2506  *   05/22/1995 JLB : Created.                                                                 *
   2507  *=============================================================================================*/
   2508 void TechnoClass::Player_Assign_Mission(MissionType mission, TARGET target, TARGET destination)
   2509 {
   2510 	if (AllowVoice) {
   2511 		if (mission == MISSION_ATTACK) {
   2512 			Response_Attack();
   2513 		} else {
   2514 			Response_Move();
   2515 		}
   2516 	}
   2517 	Queue_Mission(As_Target(), mission, target, destination);
   2518 }
   2519 
   2520 
   2521 /***********************************************************************************************
   2522  * TechnoClass::What_Action -- Determines what action to perform if object is selected.        *
   2523  *                                                                                             *
   2524  *    This routine will examine the object specified and return with the action that will      *
   2525  *    be performed if the mouse button were clicked over the object.                           *
   2526  *                                                                                             *
   2527  * INPUT:   object   -- The object that the mouse button might be clicked on.                  *
   2528  *                                                                                             *
   2529  * OUTPUT:  Returns with the action that will be performed if the object was clicked on.       *
   2530  *                                                                                             *
   2531  * WARNINGS:   none                                                                            *
   2532  *                                                                                             *
   2533  * HISTORY:                                                                                    *
   2534  *   01/19/1995 JLB : Created.                                                                 *
   2535  *   03/21/1995 JLB : Special target control for trees.                                        *
   2536  *=============================================================================================*/
   2537 ActionType TechnoClass::What_Action(ObjectClass * object) const
   2538 {
   2539 	if (object) {
   2540 
   2541 		/*
   2542 		**	Return the ACTION_SELF flag if clicking on itself. However, if this
   2543 		**	object cannot do anything special with itself, then just return with
   2544 		**	the no action flag.
   2545 		*/
   2546 		if (object == this && CurrentObject.Count() == 1 && House == PlayerPtr) {
   2547 			return(ACTION_SELF);
   2548 		}
   2549 
   2550 		//bool altdown = (Keyboard::Down(KN_LALT) || Keyboard::Down(KN_RALT));
   2551 		//bool ctrldown = (Keyboard::Down(KN_LCTRL) || Keyboard::Down(KN_RCTRL));
   2552 		//bool shiftdown = (Keyboard::Down(KN_LSHIFT) || Keyboard::Down(KN_RSHIFT));
   2553 		//Added for getting the input for special character keys from the client 
   2554 		// - 6/26/2019 JAS 
   2555 		bool altdown = DLL_Export_Get_Input_Key_State(KN_LALT);
   2556 		bool ctrldown = DLL_Export_Get_Input_Key_State(KN_LCTRL);
   2557 		bool shiftdown = DLL_Export_Get_Input_Key_State(KN_LSHIFT);
   2558 
   2559 		/*
   2560 		**	Special guard area mission is possible if both the control and the
   2561 		**	alt keys are held down.
   2562 		*/
   2563 		// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2564 		if (Is_Owned_By_Player() && ctrldown && altdown && Can_Player_Move() && Can_Player_Fire()) {
   2565 		//if (IsOwnedByPlayer && ctrldown && altdown && Can_Player_Move() && Can_Player_Fire()) {
   2566 			return(ACTION_GUARD_AREA);
   2567 		}
   2568 
   2569 		/*
   2570 		**	Special override to force a move regardless of what is occupying the location.
   2571 		*/
   2572 		if (altdown) {
   2573 			// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2574 			if (Is_Owned_By_Player() && Can_Player_Move()) {
   2575 			//if (IsOwnedByPlayer && Can_Player_Move()) {
   2576 				return(ACTION_MOVE);
   2577 			}
   2578 		}
   2579 
   2580 		/*
   2581 		**	Override so that toggled select state can be performed while the <SHIFT> key
   2582 		**	is held down.
   2583 		*/
   2584 		bool is_a_loaner = object->Is_Techno() && ((TechnoClass*)object)->IsALoaner;
   2585 		if (shiftdown) {
   2586 			// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2587 			if (!is_a_loaner) {
   2588 			//if (IsOwnedByPlayer && !IsALoaner) {
   2589 				return(ACTION_TOGGLE_SELECT);
   2590 			}
   2591 		}
   2592 
   2593 		/*
   2594 		**	If firing is possible and legal, then return this action potential.
   2595 		*/
   2596 		bool control = Keyboard::Down(KN_LCTRL) || Keyboard::Down(KN_RCTRL);
   2597 		// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2598 		if (Is_Owned_By_Player() && (ctrldown || !House->Is_Ally(object)) && (ctrldown || object->Class_Of().IsLegalTarget || (Special.IsTreeTarget && object->What_Am_I() == RTTI_TERRAIN))) {
   2599 		//if (IsOwnedByPlayer && (ctrldown || !House->Is_Ally(object)) && (ctrldown || object->Class_Of().IsLegalTarget || (Special.IsTreeTarget && object->What_Am_I() == RTTI_TERRAIN))) {
   2600 			if (Can_Player_Move() || In_Range(object, 0)) {
   2601 				// Check for anti-air capability
   2602 				if (object->What_Am_I() != RTTI_AIRCRAFT) {
   2603 					return(ACTION_ATTACK);
   2604 				} else {
   2605 					AircraftClass* aircraft = (AircraftClass*)object;
   2606 					if (*aircraft != AIRCRAFT_CARGO) {
   2607 						if (((Techno_Type_Class()->Primary != WEAPON_NONE) && BulletTypeClass::As_Reference(Weapons[Techno_Type_Class()->Primary].Fires).IsAntiAircraft) ||
   2608 							((Techno_Type_Class()->Secondary != WEAPON_NONE) && BulletTypeClass::As_Reference(Weapons[Techno_Type_Class()->Secondary].Fires).IsAntiAircraft) ||
   2609 							(aircraft->Altitude == 0)) {
   2610 							return(ACTION_ATTACK);
   2611 						}
   2612 					}
   2613 				}
   2614 			}
   2615 		}
   2616 
   2617 		/*
   2618 		**	Possibly try to select the specified object, if that is warranted.
   2619 		*/
   2620 		// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2621 		if (!Is_Weapon_Equipped() || !Is_Owned_By_Player() || object->Owner() == Owner()) {
   2622 			if ((!is_a_loaner || !Is_Owned_By_Player()) && object->Class_Of().IsSelectable && (!object->Is_Selected_By_Player() || CurrentObject.Count())) {
   2623 		//if (!Is_Weapon_Equipped() || !IsOwnedByPlayer || object->Owner() == Owner()) {
   2624 			//if ((!IsALoaner || !IsOwnedByPlayer) && object->Class_Of().IsSelectable && !object->IsSelected) {
   2625 				return(ACTION_SELECT);
   2626 			}
   2627 			return(ACTION_NONE);
   2628 		}
   2629 	}
   2630 	return(ACTION_NONE);
   2631 }
   2632 
   2633 
   2634 /***********************************************************************************************
   2635  * TechnoClass::What_Action -- Determines action to perform if cell is clicked on.             *
   2636  *                                                                                             *
   2637  *    Use this routine to determine what action will be performed if the specified cell        *
   2638  *    is clicked on. Usually this action is either a ACTION_MOVE or ACTION_NOMOVE. The action  *
   2639  *    nomove is used to perform special case checking for nearby cells if in fact the mouse    *
   2640  *    is clicked over the cell.                                                                *
   2641  *                                                                                             *
   2642  * INPUT:   cell  -- The cell to check for being clicked over.                                 *
   2643  *                                                                                             *
   2644  * OUTPUT:  Returns with the action that will occur if the cell is clicked on.                 *
   2645  *                                                                                             *
   2646  * WARNINGS:   none                                                                            *
   2647  *                                                                                             *
   2648  * HISTORY:                                                                                    *
   2649  *   01/19/1995 JLB : Created.                                                                 *
   2650  *   07/10/1995 JLB : Force fire for buildings is explicitely disabled.                        *
   2651  *=============================================================================================*/
   2652 ActionType TechnoClass::What_Action(CELL cell) const
   2653 {
   2654 	CellClass const * cellptr = &Map[cell];
   2655 	OverlayTypeClass const * optr = NULL;
   2656 
   2657 	//bool ctrldown = Keyboard::Down(KN_LCTRL) || Keyboard::Down(KN_RCTRL);
   2658 	//bool shiftdown = Keyboard::Down(KN_LSHIFT) || Keyboard::Down(KN_RSHIFT);
   2659 	//bool altdown = (Keyboard::Down(KN_LALT) || Keyboard::Down(KN_RALT));
   2660 	//Added for getting the input for special character keys from the client 
   2661 	// - 6/26/2019 JAS 
   2662 	bool altdown = DLL_Export_Get_Input_Key_State(KN_LALT);
   2663 	bool ctrldown = DLL_Export_Get_Input_Key_State(KN_LCTRL);
   2664 	bool shiftdown = DLL_Export_Get_Input_Key_State(KN_LSHIFT);
   2665 
   2666 	/*
   2667 	**	Disable recognizing the <CTRL> key forced fire option when dealing with buildings.
   2668 	*/
   2669 	if (What_Am_I() == RTTI_BUILDING) ctrldown = false;
   2670 
   2671 	if (cellptr->Overlay != OVERLAY_NONE) {
   2672 		optr = &OverlayTypeClass::As_Reference(cellptr->Overlay);
   2673 	}
   2674 
   2675 	/*
   2676 	**	Special guard area mission is possible if both the control and the
   2677 	**	alt keys are held down.
   2678 	*/
   2679 	// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2680 	if (Is_Owned_By_Player() && ctrldown && altdown && Can_Player_Move() && Can_Player_Fire()) {
   2681 	//if (IsOwnedByPlayer && ctrldown && altdown && Can_Player_Move() && Can_Player_Fire()) {
   2682 		return(ACTION_GUARD_AREA);
   2683 	}
   2684 
   2685 	// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2686 	if (Is_Owned_By_Player() && Techno_Type_Class()->Primary != WEAPON_NONE && (ctrldown || (optr && optr->IsLegalTarget))) {
   2687 	//if (IsOwnedByPlayer && Techno_Type_Class()->Primary != WEAPON_NONE && (ctrldown || (optr && optr->IsLegalTarget))) {
   2688 		WarheadTypeClass const * whead = &Warheads[BulletTypeClass::As_Reference(Weapons[Techno_Type_Class()->Primary].Fires).Warhead];
   2689 		if (!optr || (optr->IsWall && (whead->IsWallDestroyer || (whead->IsWoodDestroyer && optr->IsWooden)))) {
   2690 			if (Can_Player_Move() || In_Range(::As_Target(cell), 0)) {
   2691 				return(ACTION_ATTACK);
   2692 			}
   2693 		}
   2694 	}
   2695 
   2696 	// Changed for multiplayer. ST - 3/13/2019 5:52PM
   2697 	if (Is_Owned_By_Player() && Can_Player_Move()) {
   2698 	//if (IsOwnedByPlayer && Can_Player_Move()) {
   2699 
   2700 		/*
   2701 		**	Special override to force a move regardless of what is occupying the location.
   2702 		*/
   2703 		if (shiftdown) {
   2704 			return(ACTION_MOVE);
   2705 		}
   2706 
   2707 		/*
   2708 		**	If the object can enter the cell specified, then allow
   2709 		**	movement to it.
   2710 		*/
   2711 		if (Can_Enter_Cell(cell) <= MOVE_CLOAK) {
   2712 			return(ACTION_MOVE);
   2713 		}
   2714 		return(ACTION_NOMOVE);
   2715 	}
   2716 	return(ACTION_NONE);
   2717 }
   2718 
   2719 
   2720 /***********************************************************************************************
   2721  * TechnoClass::Can_Player_Move -- Determines if the object can move be moved by player.       *
   2722  *                                                                                             *
   2723  *    Use this routine to determine whether a movement order can be given to this object.      *
   2724  *                                                                                             *
   2725  * INPUT:   none                                                                               *
   2726  *                                                                                             *
   2727  * OUTPUT:  bool; Can this object be given a movement order by the player?                     *
   2728  *                                                                                             *
   2729  * WARNINGS:   none                                                                            *
   2730  *                                                                                             *
   2731  * HISTORY:                                                                                    *
   2732  *   01/19/1995 JLB : Created.                                                                 *
   2733  *=============================================================================================*/
   2734 bool TechnoClass::Can_Player_Move(void) const
   2735 {
   2736 	return(PlayerPtr == House);
   2737 }
   2738 
   2739 
   2740 /***********************************************************************************************
   2741  * TechnoClass::Can_Player_Fire -- Determines if the player can give this object a fire order. *
   2742  *                                                                                             *
   2743  *    Call this routine to determine if this object can be given a fire order by the player.   *
   2744  *    Such objects will affect the mouse cursor accordingly -- usually causes the targeting    *
   2745  *    cursor to appear.                                                                        *
   2746  *                                                                                             *
   2747  * INPUT:   none                                                                               *
   2748  *                                                                                             *
   2749  * OUTPUT:  bool; Can this object be given firing orders by the player?                        *
   2750  *                                                                                             *
   2751  * WARNINGS:   none                                                                            *
   2752  *                                                                                             *
   2753  * HISTORY:                                                                                    *
   2754  *   01/23/1995 JLB : Created.                                                                 *
   2755  *=============================================================================================*/
   2756 bool TechnoClass::Can_Player_Fire(void) const
   2757 {
   2758 	if (House->IsHuman && Is_Techno() && Techno_Type_Class()->Primary != WEAPON_NONE) {
   2759 		return(true);
   2760 	}
   2761 	return(false);
   2762 }
   2763 
   2764 
   2765 /***********************************************************************************************
   2766  * TechnoClass::Is_Weapon_Equipped -- Determines if this object has a combat weapon.          *
   2767  *                                                                                             *
   2768  *    Use this routine to determine if this object is equipped with a combat weapon. Such      *
   2769  *    determination is used by the AI system to gauge the threat potential of the object.      *
   2770  *                                                                                             *
   2771  * INPUT:   none                                                                               *
   2772  *                                                                                             *
   2773  * OUTPUT:  bool; Is this object equipped with a combat weapon?                                *
   2774  *                                                                                             *
   2775  * WARNINGS:   none                                                                            *
   2776  *                                                                                             *
   2777  * HISTORY:                                                                                    *
   2778  *   01/23/1995 JLB : Created.                                                                 *
   2779  *=============================================================================================*/
   2780 bool TechnoClass::Is_Weapon_Equipped(void) const
   2781 {
   2782 	return(Techno_Type_Class()->Primary != WEAPON_NONE);
   2783 }
   2784 
   2785 
   2786 /***********************************************************************************************
   2787  * TechnoClass::Can_Repair -- Determines if the object can and should be repaired.             *
   2788  *                                                                                             *
   2789  *    Use this routine to determine if the specified object is a candidate for repair. In      *
   2790  *    order to qualify, the object must be allowed to be repaired (in theory) and it must      *
   2791  *    be below full strength. If these conditions are met, then it can be repaired.            *
   2792  *                                                                                             *
   2793  * INPUT:   none                                                                               *
   2794  *                                                                                             *
   2795  * OUTPUT:  bool; May this unit be repaired? A return value of false may mean that the object  *
   2796  *                is not allowed to be repaired, or it might be full strength already.         *
   2797  *                                                                                             *
   2798  * WARNINGS:   none                                                                            *
   2799  *                                                                                             *
   2800  * HISTORY:                                                                                    *
   2801  *   01/23/1995 JLB : Created.                                                                 *
   2802  *=============================================================================================*/
   2803 bool TechnoClass::Can_Repair(void) const
   2804 {
   2805 	/*
   2806 	**	Temporary hack to disable repair cursor over non-buildings.
   2807 	*/
   2808 	if (What_Am_I() == RTTI_UNIT || What_Am_I() == RTTI_INFANTRY || What_Am_I() == RTTI_AIRCRAFT) {
   2809 		return(false);
   2810 	}
   2811 	return(Techno_Type_Class()->IsRepairable && Strength != Class_Of().MaxStrength);
   2812 }
   2813 
   2814 
   2815 /***********************************************************************************************
   2816  * TechnoClass::Weapon_Range -- Determines the maximum range for the weapon.                   *
   2817  *                                                                                             *
   2818  *    Use this routine to determine the maximum range for the weapon indicated.                *
   2819  *                                                                                             *
   2820  * INPUT:   which -- Which weapon to use when determining the range. 0=primary, 1=secondary.   *
   2821  *                                                                                             *
   2822  * OUTPUT:  Returns with the range of the weapon (in leptons).                                 *
   2823  *                                                                                             *
   2824  * WARNINGS:   none                                                                            *
   2825  *                                                                                             *
   2826  * HISTORY:                                                                                    *
   2827  *   03/19/1995 JLB : Created.                                                                 *
   2828  *=============================================================================================*/
   2829 int TechnoClass::Weapon_Range(int which) const
   2830 {
   2831 	WeaponType weapon = WEAPON_NONE;
   2832 	TechnoTypeClass const & ttype = *Techno_Type_Class();
   2833 
   2834 	switch (which) {
   2835 		case 0:
   2836 			weapon = ttype.Primary;
   2837 			break;
   2838 
   2839 		case 1:
   2840 			weapon = ttype.Secondary;
   2841 			break;
   2842 	}
   2843 	if (weapon != WEAPON_NONE) {
   2844 		if (weapon == WEAPON_NIKE && GameToPlay == GAME_NORMAL) {
   2845 			return(Weapons[weapon].Range*2);
   2846 		}
   2847 		return(Weapons[weapon].Range);
   2848 	}
   2849 	return(0);
   2850 }
   2851 
   2852 
   2853 /***************************************************************************
   2854  * TechnoClass::Override_Mission -- temporarily overides a units mission   *
   2855  *                                                                         *
   2856  *                                                                         *
   2857  *                                                                         *
   2858  * INPUT:      MissionType mission - the mission we want to overide        *
   2859  *               TARGET      tarcom  - the new target we want to overide   *
   2860  *               TARGET      navcom  - the new navigation point to overide *
   2861  *                                                                         *
   2862  * OUTPUT:      none                                                       *
   2863  *                                                                         *
   2864  * WARNINGS:   If a mission is already overidden, the current mission is   *
   2865  *               just re-assigned.                                         *
   2866  *                                                                         *
   2867  * HISTORY:                                                                *
   2868  *   04/28/1995 PWG : Created.                                             *
   2869  *=========================================================================*/
   2870 void TechnoClass::Override_Mission(MissionType mission, TARGET tarcom, TARGET navcom)
   2871 {
   2872 	SuspendedTarCom = TarCom;
   2873 	RadioClass::Override_Mission(mission, tarcom, navcom);
   2874 	Assign_Target(tarcom);
   2875 }
   2876 
   2877 
   2878 /***************************************************************************
   2879  * TechnoClass::Restore_Mission -- Restores an overidden mission           *
   2880  *                                                                         *
   2881  * INPUT:		none                                                        *
   2882  *                                                                         *
   2883  * OUTPUT:     none                                                        *
   2884  *                                                                         *
   2885  * WARNINGS:   none                                                        *
   2886  *                                                                         *
   2887  * HISTORY:                                                                *
   2888  *   04/28/1995 PWG : Created.                                             *
   2889  *=========================================================================*/
   2890 bool TechnoClass::Restore_Mission(void)
   2891 {
   2892 	if (RadioClass::Restore_Mission()) {
   2893 		Assign_Target(SuspendedTarCom);
   2894 		return(true);
   2895 	}
   2896 	return(false);
   2897 }
   2898 
   2899 
   2900 /***********************************************************************************************
   2901  * TechnoClass::Captured -- Handles capturing this object.                                     *
   2902  *                                                                                             *
   2903  *    This routine is called when this object gets captured by the house specified. It handles *
   2904  *    removing this object from any targeting computers and then changes the ownership of      *
   2905  *    the object to the new house.                                                             *
   2906  *                                                                                             *
   2907  * INPUT:   newowner -- Pointer to the house that is now the new owner.                        *
   2908  *                                                                                             *
   2909  * OUTPUT:  Was the object captured? Failure would mean that it is already under control of    *
   2910  *          the house specified.                                                               *
   2911  *                                                                                             *
   2912  * WARNINGS:   none                                                                            *
   2913  *                                                                                             *
   2914  * HISTORY:                                                                                    *
   2915  *   05/08/1995 JLB : Created.                                                                 *
   2916  *=============================================================================================*/
   2917 bool TechnoClass::Captured(HouseClass * newowner)
   2918 {
   2919 	if (newowner != House) {
   2920 
   2921 		/*
   2922 		**	Capture attempt springs any "entered" trigger. The entered trigger
   2923 		**	occurs first since there may be a special trigger attached to this
   2924 		**	object that flags a capture as a win and a destroy as a loss. This
   2925 		**	order is necessary because the object is recorded as a kill as well.
   2926 		*/
   2927 		if (Trigger /*&& Trigger->House == newowner->Class->House*/) {
   2928 			Trigger->Spring(EVENT_PLAYER_ENTERED, this);
   2929 		}
   2930 
   2931 		/*
   2932 		**	Record this as a kill.
   2933 		*/
   2934 		Record_The_Kill(NULL);
   2935 
   2936 #ifdef USE_RA_AI
   2937 		//
   2938 		// Added for RA AI in TD. ST - 7/26/2019 9:12AM
   2939 		//
   2940 		House->Tracking_Remove(this);
   2941 		newowner->Tracking_Add(this);
   2942 #endif // USE_RA_AI
   2943 
   2944 		/*
   2945 		**	Special kill record logic for capture process.
   2946 		*/
   2947 		switch (What_Am_I()) {
   2948 			case RTTI_BUILDING:
   2949 				if (newowner) newowner->BuildingsKilled[Owner()]++;
   2950 				break;
   2951 
   2952 			case RTTI_AIRCRAFT:
   2953 			case RTTI_INFANTRY:
   2954 			case RTTI_UNIT:
   2955 				if (newowner) newowner->UnitsKilled[Owner()]++;
   2956 				break;
   2957 
   2958 			default:
   2959 				break;
   2960 		}
   2961 		House->WhoLastHurtMe = newowner->Class->House;
   2962 
   2963 		/*
   2964 		**	Remove from targeting computers.
   2965 		*/
   2966 		Detach_All(false);
   2967 
   2968 #ifdef NEVER
   2969 		/*
   2970 		**	Break off any radio contact.
   2971 		*/
   2972 		Transmit_Message(RADIO_OVER_OUT);
   2973 #endif
   2974 
   2975 		/*
   2976 		**	Change ownership now.
   2977 		*/
   2978 		House = newowner;
   2979 		IsOwnedByPlayer = (House == PlayerPtr);
   2980 
   2981 		return(true);
   2982 	}
   2983 	return(false);
   2984 }
   2985 
   2986 
   2987 /***********************************************************************************************
   2988  * TechnoClass::Take_Damage -- Records damage assessed to this object.                         *
   2989  *                                                                                             *
   2990  *    This routine is called when this object has taken damage. It handles recording whether   *
   2991  *    this object has been destroyed. If it has, then mark the appropriate kill records as     *
   2992  *    necessary.                                                                               *
   2993  *                                                                                             *
   2994  * INPUT:                                                                                      *
   2995  *                                                                                             *
   2996  * OUTPUT:                                                                                     *
   2997  *                                                                                             *
   2998  * WARNINGS:                                                                                   *
   2999  *                                                                                             *
   3000  * HISTORY:                                                                                    *
   3001  *   06/20/1995 JLB : Created.                                                                 *
   3002  *=============================================================================================*/
   3003 ResultType TechnoClass::Take_Damage(int & damage, int distance, WarheadType warhead, TechnoClass * source)
   3004 {
   3005 	/*
   3006 	**	Adjust damage according to house override armor value.
   3007 	*/
   3008 	if (damage > 0) {
   3009 		damage = damage * House->ArmorBias;
   3010 	}
   3011 
   3012 	ResultType result = ObjectClass::Take_Damage(damage, distance, warhead, source);
   3013 
   3014 	switch (result) {
   3015 		case RESULT_DESTROYED:
   3016 			Transmit_Message(RADIO_OVER_OUT);
   3017 			Stun();
   3018 
   3019 			/*
   3020 			** May trigger an achievement. ST - 11/14/2019 1:56PM
   3021 			*/
   3022 			if (GameToPlay == GAME_NORMAL && !House->IsHuman && source && source->House && source->House->IsHuman) {
   3023 				TechnoTypeClass const *object_type = Techno_Type_Class();
   3024 				if (object_type) {
   3025 					RTTIType what = What_Am_I();
   3026 					if (what == RTTI_AIRCRAFT || what == RTTI_INFANTRY || what == RTTI_UNIT) {
   3027 						On_Achievement_Event(source->House, "UNIT_DESTROYED", object_type->IniName);
   3028 					}
   3029 				}			
   3030 			}
   3031 			break;
   3032 
   3033 		/*
   3034 		**	If some damage was received and this object is cloaked, shimmer
   3035 		**	the cloak a bit.
   3036 		*/
   3037 		default:
   3038 			if (source && !House->Is_Ally(source)) {
   3039 				IsTickedOff = true;
   3040 			}
   3041 			Do_Shimmer();
   3042 			break;
   3043 
   3044 		case RESULT_NONE:
   3045 			break;
   3046 	}
   3047 	return(result);
   3048 }
   3049 
   3050 
   3051 /***********************************************************************************************
   3052  * TechnoTypeClass::Max_Passengers -- Fetches the maximum passengers allowed.                  *
   3053  *                                                                                             *
   3054  *    This routine will return with the maximum number of passengers allowed in this           *
   3055  *    transport. This typically applies to APCs and possibley transport helicopters.           *
   3056  *                                                                                             *
   3057  * INPUT:   none                                                                               *
   3058  *                                                                                             *
   3059  * OUTPUT:  Returns with the number of passengers this transport can carry.                    *
   3060  *                                                                                             *
   3061  * WARNINGS:   none                                                                            *
   3062  *                                                                                             *
   3063  * HISTORY:                                                                                    *
   3064  *   06/20/1995 JLB : Created.                                                                 *
   3065  *=============================================================================================*/
   3066 int TechnoTypeClass::Max_Passengers(void) const
   3067 {
   3068 	if (IsTransporter) {
   3069 		return(5);
   3070 	}
   3071 	return(0);
   3072 }
   3073 
   3074 
   3075 /***********************************************************************************************
   3076  * TechnoClass::Record_The_Kill -- Records the death of this object.                           *
   3077  *                                                                                             *
   3078  *    This routine is used to record the death of this object. It will handle updating the     *
   3079  *    owner house with the kill record as well as springing any trigger events associated with *
   3080  *    this object's death.                                                                     *
   3081  *                                                                                             *
   3082  * INPUT:   source   -- Pointer to the source of this object's death (if there is a source).   *
   3083  *                                                                                             *
   3084  * OUTPUT:  none                                                                               *
   3085  *                                                                                             *
   3086  * WARNINGS:   none                                                                            *
   3087  *                                                                                             *
   3088  * HISTORY:                                                                                    *
   3089  *   07/08/1995 JLB : Created.                                                                 *
   3090  *   08/23/1995 JLB : Building loss is only counted if it received damage.                     *
   3091  *=============================================================================================*/
   3092 void TechnoClass::Record_The_Kill(TechnoClass * source)
   3093 {
   3094 	/*
   3095 	**	Handle any trigger event associated with this object.
   3096 	*/
   3097 	if (Trigger && source) Trigger->Spring(EVENT_ATTACKED, this);
   3098 
   3099 	if (Trigger && source) Trigger->Spring(EVENT_DISCOVERED, this);
   3100 
   3101 	if (Trigger) Trigger->Spring(EVENT_DESTROYED, this);
   3102 
   3103 	if (source) {
   3104 		/*
   3105 		**	Call the explicity cast versions of the Made_A_Kill function. This
   3106 		**	is necessary because we don't want to add a virtual function to the
   3107 		**	CrewClass. Doing so would complicate the save/load process.
   3108 		*/
   3109 		switch (source->What_Am_I()) {
   3110 			case RTTI_INFANTRY:
   3111 				((InfantryClass *)source)->Made_A_Kill();
   3112 				break;
   3113 
   3114 			case RTTI_UNIT:
   3115 				((UnitClass *)source)->Made_A_Kill();
   3116 				break;
   3117 
   3118 			case RTTI_BUILDING:
   3119 				((BuildingClass *)source)->Made_A_Kill();
   3120 				break;
   3121 
   3122 			case RTTI_AIRCRAFT:
   3123 				((AircraftClass *)source)->Made_A_Kill();
   3124 				break;
   3125 		}
   3126 
   3127 		House->WhoLastHurtMe = source->Owner();
   3128 	}
   3129 
   3130 	switch (What_Am_I()) {
   3131 		case RTTI_BUILDING:
   3132 			if ( ((BuildingClass *)this)->WhoLastHurtMe != HOUSE_NONE) {
   3133 				House->BuildingsLost++;
   3134 			}
   3135 			if (source){
   3136 				if (GameToPlay == GAME_INTERNET){
   3137 					source->House->DestroyedBuildings->Increment_Unit_Total( ((BuildingClass*)this)->Class->Type );
   3138 				}
   3139 				source->House->BuildingsKilled[Owner()]++;
   3140 			}
   3141 
   3142 			/*
   3143 			** If the map is displaying the multiplayer player names & their
   3144 			** # of kills, tell it to redraw.
   3145 			*/
   3146 			if (Map.Is_Player_Names()) {
   3147 				Map.Player_Names(true);
   3148 			}
   3149 			break;
   3150 
   3151 
   3152 		case RTTI_AIRCRAFT:
   3153 			House->UnitsLost++;
   3154 			if (source){
   3155 				if (GameToPlay == GAME_INTERNET){
   3156 					source->House->DestroyedAircraft->Increment_Unit_Total( ((AircraftClass*)this)->Class->Type );
   3157 				}
   3158 				source->House->UnitsKilled[Owner()]++;
   3159 			}
   3160 			/*
   3161 			** If the map is displaying the multiplayer player names & their
   3162 			** # of kills, tell it to redraw.
   3163 			*/
   3164 			if (Map.Is_Player_Names()) {
   3165 				Map.Player_Names(true);
   3166 			}
   3167 			break;
   3168 
   3169 
   3170 		case RTTI_INFANTRY:
   3171 			House->UnitsLost++;
   3172 			if (source){
   3173 				if (GameToPlay == GAME_INTERNET){
   3174 					source->House->DestroyedInfantry->Increment_Unit_Total( ((InfantryClass*)this)->Class->Type );
   3175 				}
   3176 				source->House->UnitsKilled[Owner()]++;
   3177 			}
   3178 			/*
   3179 			** If the map is displaying the multiplayer player names & their
   3180 			** # of kills, tell it to redraw.
   3181 			*/
   3182 			if (Map.Is_Player_Names()) {
   3183 				Map.Player_Names(true);
   3184 			}
   3185 			break;
   3186 
   3187 
   3188 		case RTTI_UNIT:
   3189 			House->UnitsLost++;
   3190 			if (source){
   3191 				if (GameToPlay == GAME_INTERNET){
   3192 					source->House->DestroyedUnits->Increment_Unit_Total( ((UnitClass*)this)->Class->Type );
   3193 				}
   3194 				source->House->UnitsKilled[Owner()]++;
   3195 			}
   3196 
   3197 			/*
   3198 			** If the map is displaying the multiplayer player names & their
   3199 			** # of kills, tell it to redraw.
   3200 			*/
   3201 			if (Map.Is_Player_Names()) {
   3202 				Map.Player_Names(true);
   3203 			}
   3204 			break;
   3205 
   3206 		default:
   3207 			break;
   3208 	}
   3209 }
   3210 
   3211 
   3212 /***********************************************************************************************
   3213  * TechnoClass::Nearby_Location -- Radiates outward looking for clear cell nearby.             *
   3214  *                                                                                             *
   3215  *    This routine is used to find a nearby location from center of this object. It can lean   *
   3216  *    toward finding a location closest to an optional object.                                 *
   3217  *                                                                                             *
   3218  * INPUT:   object   -- Optional object that the finding algorithm will try to find a close    *
   3219  *                      spot to.                                                               *
   3220  *                                                                                             *
   3221  * OUTPUT:  Returns with the cell that is closest to this object.                              *
   3222  *                                                                                             *
   3223  * WARNINGS:   none                                                                            *
   3224  *                                                                                             *
   3225  * HISTORY:                                                                                    *
   3226  *   07/06/1995 JLB : Created.                                                                 *
   3227  *=============================================================================================*/
   3228 CELL TechnoClass::Nearby_Location(TechnoClass const * ) const
   3229 {
   3230 	/*
   3231 	**	Radiate outward from the object's location, looking for the best
   3232 	**	target.
   3233 	*/
   3234 	CELL best = 0;
   3235 	CELL cell = Coord_Cell(Center_Coord());
   3236 	for (int radius = 0; radius < MAP_CELL_W/2; radius++) {
   3237 
   3238 		/*
   3239 		**	Scan the top and bottom rows of the "box".
   3240 		*/
   3241 		for (int x = -radius; x <= radius; x++) {
   3242 			CELL newcell = cell + XY_Cell(x, -radius);
   3243 			if (Map.In_Radar(newcell) && Map[newcell].Is_Generally_Clear()) {
   3244 				best = newcell;
   3245 			}
   3246 
   3247 			newcell = cell + XY_Cell(x, radius);
   3248 			if (Map.In_Radar(newcell) && Map[newcell].Is_Generally_Clear()) {
   3249 				best = newcell;
   3250 			}
   3251 		}
   3252 
   3253 		/*
   3254 		**	Scan the left and right columns of the "box".
   3255 		*/
   3256 		for (int y = -(radius-1); y < radius; y++) {
   3257 			CELL newcell = cell + XY_Cell(-radius, y);
   3258 			if (Map.In_Radar(newcell) && Map[newcell].Is_Generally_Clear()) {
   3259 				best = newcell;
   3260 			}
   3261 
   3262 			newcell = cell + XY_Cell(radius, y);
   3263 			if (Map.In_Radar(newcell) && Map[newcell].Is_Generally_Clear()) {
   3264 				best = newcell;
   3265 			}
   3266 		}
   3267 
   3268 		if (best) break;
   3269 	}
   3270 
   3271 	return(best);
   3272 }
   3273 
   3274 
   3275 /***********************************************************************************************
   3276  * TechnoClass::Do_Uncloak -- Cause the stealth tank to uncloak.                               *
   3277  *                                                                                             *
   3278  *    This routine will start the stealth tank to uncloak.                                     *
   3279  *                                                                                             *
   3280  * INPUT:   none                                                                               *
   3281  *                                                                                             *
   3282  * OUTPUT:  none                                                                               *
   3283  *                                                                                             *
   3284  * WARNINGS:   none                                                                            *
   3285  *                                                                                             *
   3286  * HISTORY:                                                                                    *
   3287  *   05/08/1995 JLB : Created.                                                                 *
   3288  *=============================================================================================*/
   3289 void TechnoClass::Do_Uncloak(void)
   3290 {
   3291 	if (IsCloakable && (Cloak == CLOAKED || Cloak == CLOAKING)) {
   3292 		Sound_Effect(VOC_CLOAK, Coord);
   3293 		Cloak = UNCLOAKING;
   3294 		CloakingDevice.Set_Stage(0);
   3295 		CloakingDevice.Set_Rate(1);
   3296 	}
   3297 }
   3298 
   3299 
   3300 /***********************************************************************************************
   3301  * TechnoClass::Do_Cloak -- Start the object into cloaking stage.                              *
   3302  *                                                                                             *
   3303  *    This routine will start the object into its cloaking state.                              *
   3304  *                                                                                             *
   3305  * INPUT:   none                                                                               *
   3306  *                                                                                             *
   3307  * OUTPUT:  none                                                                               *
   3308  *                                                                                             *
   3309  * WARNINGS:   none                                                                            *
   3310  *                                                                                             *
   3311  * HISTORY:                                                                                    *
   3312  *   07/08/1995 JLB : Created.                                                                 *
   3313  *=============================================================================================*/
   3314 void TechnoClass::Do_Cloak(void)
   3315 {
   3316 	if (IsCloakable && (Cloak == UNCLOAKED || Cloak == UNCLOAKING)) {
   3317 		Sound_Effect(VOC_CLOAK, Coord);
   3318 		Detach_All(false);
   3319 		Cloak = CLOAKING;
   3320 		CloakingDevice.Set_Stage(0);
   3321 		CloakingDevice.Set_Rate(1);
   3322 	}
   3323 }
   3324 
   3325 
   3326 /***********************************************************************************************
   3327  * TechnoClass::Do_Shimmer -- Causes this object to shimmer if it is cloaked.                  *
   3328  *                                                                                             *
   3329  *    This routine is called when this object should shimmer. If the object is cloaked, then   *
   3330  *    a shimmering effect (partial decloak) occurs. For objects that are not cloaked, no       *
   3331  *    affect occurs.                                                                           *
   3332  *                                                                                             *
   3333  * INPUT:   none                                                                               *
   3334  *                                                                                             *
   3335  * OUTPUT:  none                                                                               *
   3336  *                                                                                             *
   3337  * WARNINGS:   none                                                                            *
   3338  *                                                                                             *
   3339  * HISTORY:                                                                                    *
   3340  *   07/29/1995 JLB : Created.                                                                 *
   3341  *=============================================================================================*/
   3342 void TechnoClass::Do_Shimmer(void)
   3343 {
   3344 	if (IsCloakable && Cloak == CLOAKED) {
   3345 		Cloak = CLOAKING;
   3346 		CloakingDevice.Set_Stage(MAX_UNCLOAK_STAGE/2);
   3347 		CloakingDevice.Set_Rate(1);
   3348 	}
   3349 }
   3350 
   3351 
   3352 /***********************************************************************************************
   3353  * TechnoClass::Visual_Character -- Determine the visual character of the object.              *
   3354  *                                                                                             *
   3355  *    This routine will determine how this object should be drawn. Typically, this is the      *
   3356  *    unmodified visible state, but cloaked objects have a different character.                *
   3357  *                                                                                             *
   3358  * INPUT:   raw   -- Should the check be based on the unmodified cloak condition of the        *
   3359  *                   object? If false, then an object owned by the player will never become    *
   3360  *                   completely invisible.                                                     *
   3361  *                                                                                             *
   3362  * OUTPUT:  Returns with the visual character to use when displaying this object.              *
   3363  *                                                                                             *
   3364  * WARNINGS:   none                                                                            *
   3365  *                                                                                             *
   3366  * HISTORY:                                                                                    *
   3367  *   07/07/1995 JLB : Created.                                                                 *
   3368  *=============================================================================================*/
   3369 VisualType TechnoClass::Visual_Character(bool raw)
   3370 {
   3371 	/*
   3372 	**	When uncloaked or in map editor mode, always draw the object normally.
   3373 	*/
   3374 	if (Cloak == UNCLOAKED || Debug_Map) return(VISUAL_NORMAL);
   3375 
   3376 	/*
   3377 	**	A cloaked unit will not be visible at all unless it is owned
   3378 	**	by the player.
   3379 	*/
   3380 	if (Cloak == CLOAKED) {
   3381 		// Changed for multiplayer. Not needed except to test in the old renderer. ST - 3/13/2019 5:56PM
   3382 		if (!raw && Is_Owned_By_Player()) return(VISUAL_SHADOWY);
   3383 		//if (!raw && IsOwnedByPlayer) return(VISUAL_SHADOWY);
   3384 		return(VISUAL_HIDDEN);
   3385 	}
   3386 
   3387 	int stage = CloakingDevice.Fetch_Stage();
   3388 	if (Cloak == UNCLOAKING) stage = MAX_UNCLOAK_STAGE - stage;
   3389 	if (stage <= 0) {
   3390 		return(VISUAL_NORMAL);
   3391 	}
   3392 
   3393 	stage = Cardinal_To_Fixed(MAX_UNCLOAK_STAGE, stage);
   3394 
   3395 	if (stage < 0x0040) return(VISUAL_INDISTINCT);
   3396 	if (stage < 0x0080) return(VISUAL_DARKEN);
   3397 	if (stage < 0x00C0) return(VISUAL_SHADOWY);
   3398 	if (!raw && Is_Owned_By_Player()) return(VISUAL_SHADOWY);		// Changed for multiplayer. Not needed except to test in the old renderer. ST - 3/13/2019 5:56PM
   3399 	//if (!raw && IsOwnedByPlayer) return(VISUAL_SHADOWY);
   3400 	if (stage < 0x00FF) return(VISUAL_RIPPLE);
   3401 	return(VISUAL_HIDDEN);
   3402 }
   3403 
   3404 
   3405 /***********************************************************************************************
   3406  * TechnoClass::Techno_Draw_Object -- General purpose draw object routine.                     *
   3407  *                                                                                             *
   3408  *    This routine is used to draw the object. It will handle any remapping or cloaking        *
   3409  *    effects required. This logic is isolated here since all techno object share the same     *
   3410  *    render logic when it comes to remapping and cloaking.                                    *
   3411  *                                                                                             *
   3412  * INPUT:   shapefile   -- Pointer to the shape file that the shape will be drawn from.        *
   3413  *                                                                                             *
   3414  *          shapenum    -- The shape number of the object in the file to use.                  *
   3415  *                                                                                             *
   3416  *          x,y         -- Center pixel coordinate to use for rendering this object.           *
   3417  *                                                                                             *
   3418  *          window      -- The clipping window to use when rendering.                          *
   3419  *                                                                                             *
   3420  * OUTPUT:  none                                                                               *
   3421  *                                                                                             *
   3422  * WARNINGS:   none                                                                            *
   3423  *                                                                                             *
   3424  * HISTORY:                                                                                    *
   3425  *   07/08/1995 JLB : Created.                                                                 *
   3426  *=============================================================================================*/
   3427 void TechnoClass::Techno_Draw_Object(void const * shapefile, int shapenum, int x, int y, WindowNumberType window)
   3428 {
   3429 	if (shapefile) {
   3430 		VisualType visual = Visual_Character();
   3431 		void const * remap = Remap_Table();
   3432 
   3433 		// Server still needs to "render" hidden objects to the virtual window, so objects get created properly - SKY
   3434 		if ((visual == VISUAL_HIDDEN) && (window == WINDOW_VIRTUAL)) {
   3435 			visual = VISUAL_SHADOWY;
   3436 		}
   3437 
   3438 		if (visual != VISUAL_HIDDEN && visual != VISUAL_RIPPLE) {
   3439 			if (visual == VISUAL_SHADOWY) {
   3440 				CC_Draw_Shape(this, shapefile, shapenum, x, y, window, SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING|SHAPE_PREDATOR, NULL, Map.FadingShade);
   3441 			} else {
   3442 				CC_Draw_Shape(this, shapefile, shapenum, x, y, window, SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING|SHAPE_GHOST, remap, Map.UnitShadow);
   3443 			}
   3444 			if (visual == VISUAL_DARKEN) {
   3445 				CC_Draw_Shape(this, shapefile, shapenum, x, y, window, SHAPE_PREDATOR|SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING, remap, Map.FadingShade);
   3446 			}
   3447 		}
   3448 		if (visual != VISUAL_NORMAL && visual != VISUAL_HIDDEN) {
   3449 			CC_Draw_Shape(this, shapefile, shapenum, x, y, window, SHAPE_PREDATOR|SHAPE_CENTER|SHAPE_WIN_REL);
   3450 		}
   3451 	}
   3452 }
   3453 
   3454 
   3455 
   3456 /***********************************************************************************************
   3457  * TechnoClass::Techno_Draw_Object_Virtual -- Draw object with virtual window support          *
   3458  *                                                                                             *
   3459  * INPUT:   shapefile   -- Pointer to the shape file that the shape will be drawn from.        *
   3460  *                                                                                             *
   3461  *          shapenum    -- The shape number of the object in the file to use.                  *
   3462  *                                                                                             *
   3463  *          x,y         -- Center pixel coordinate to use for rendering this object.           *
   3464  *                                                                                             *
   3465  *          window      -- The clipping window to use when rendering.                          *
   3466  *                                                                                             *
   3467  *          shape_name  -- The name of the shapefile                                           *
   3468  *                                                                                             *
   3469  * OUTPUT:  none                                                                               *
   3470  *                                                                                             *
   3471  * WARNINGS:   none                                                                            *
   3472  *                                                                                             *
   3473  * HISTORY:                                                                                    *
   3474  *   6/20/2019 1:31PM - ST : Created.                                                          *
   3475  *=============================================================================================*/
   3476 void TechnoClass::Techno_Draw_Object_Virtual(void const * shapefile, int shapenum, int x, int y, WindowNumberType window, const char *shape_name)
   3477 {
   3478 	if (shape_name == NULL || *shape_name == 0) {
   3479 		/*
   3480 		** If there's no override shape name, then call the regular draw
   3481 		*/
   3482 		Techno_Draw_Object(shapefile, shapenum, x, y, window);
   3483 		return;
   3484 	}
   3485 	
   3486 	if (shapefile) {
   3487 		VisualType visual = Visual_Character();
   3488 		void const * remap = Remap_Table();
   3489 
   3490 		// Server still needs to "render" hidden objects to the virtual window, so objects get created properly - SKY
   3491 		if ((visual == VISUAL_HIDDEN) && (window == WINDOW_VIRTUAL)) {
   3492 			visual = VISUAL_SHADOWY;
   3493 		}
   3494 
   3495 		if (visual != VISUAL_HIDDEN && visual != VISUAL_RIPPLE) {
   3496 			if (visual == VISUAL_SHADOWY) {
   3497 				CC_Draw_Shape(this, shape_name, shapefile, shapenum, x, y, window, SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING|SHAPE_PREDATOR, NULL, Map.FadingShade);
   3498 			} else {
   3499 				CC_Draw_Shape(this, shape_name, shapefile, shapenum, x, y, window, SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING|SHAPE_GHOST, remap, Map.UnitShadow);
   3500 			}
   3501 			if (visual == VISUAL_DARKEN) {
   3502 				CC_Draw_Shape(this, shape_name, shapefile, shapenum, x, y, window, SHAPE_PREDATOR|SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING, remap, Map.FadingShade);
   3503 			}
   3504 		}
   3505 		if (visual != VISUAL_NORMAL && visual != VISUAL_HIDDEN) {
   3506 			CC_Draw_Shape(this, shape_name, shapefile, shapenum, x, y, window, SHAPE_PREDATOR|SHAPE_CENTER|SHAPE_WIN_REL);
   3507 		}
   3508 	}
   3509 }
   3510 
   3511 
   3512 
   3513 /***********************************************************************************************
   3514  * TechnoClass::Remap_Table -- Fetches the appropriate remap table to use.                     *
   3515  *                                                                                             *
   3516  *    This routine is used to fetch the appropriate remap table to use for this object.        *
   3517  *                                                                                             *
   3518  * INPUT:   none                                                                               *
   3519  *                                                                                             *
   3520  * OUTPUT:  Returns with a pointer to the remap table to use for this object.                  *
   3521  *                                                                                             *
   3522  * WARNINGS:   none                                                                            *
   3523  *                                                                                             *
   3524  * HISTORY:                                                                                    *
   3525  *   07/08/1995 JLB : Created.                                                                 *
   3526  *=============================================================================================*/
   3527 void const * TechnoClass::Remap_Table(void)
   3528 {
   3529 	return(House->Remap_Table(IsBlushing, true));
   3530 }
   3531 
   3532 
   3533 /***********************************************************************************************
   3534  * TechnoClass::Detach -- Handles removal of target from tracking system.                      *
   3535  *                                                                                             *
   3536  *    This routine is called when the specified object is about to be removed from the game    *
   3537  *    system. The target object is removed from any tracking computers that this object may    *
   3538  *    have.                                                                                    *
   3539  *                                                                                             *
   3540  * INPUT:   target   -- The target object (as a target value) that is being removed from the   *
   3541  *                      game.                                                                  *
   3542  *                                                                                             *
   3543  *          all      -- Is the target about to die? A false value might indicate that the      *
   3544  *                      object is merely cloaking. In such a case, radio contact will not      *
   3545  *                      be affected.                                                           *
   3546  *                                                                                             *
   3547  * OUTPUT:  none                                                                               *
   3548  *                                                                                             *
   3549  * WARNINGS:   none                                                                            *
   3550  *                                                                                             *
   3551  * HISTORY:                                                                                    *
   3552  *   07/29/1995 JLB : Created.                                                                 *
   3553  *=============================================================================================*/
   3554 void TechnoClass::Detach(TARGET target, bool all)
   3555 {
   3556 	RadioClass::Detach(target, all);
   3557 
   3558 	if (SuspendedMission != MISSION_NONE && SuspendedTarCom == target) {
   3559 		SuspendedMission = MISSION_NONE;
   3560 		SuspendedTarCom = TARGET_NONE;
   3561 	}
   3562 
   3563 	/*
   3564 	**	If the targeting computer is assigned to the target, then the targeting
   3565 	**	computer must be cleared.
   3566 	*/
   3567 	if (TarCom == target) {
   3568 		Assign_Target(TARGET_NONE);
   3569 		Restore_Mission();
   3570 	}
   3571 
   3572 	/*
   3573 	**	If it is in radio contact with another object, then that radio contact
   3574 	**	must be broken.
   3575 	*/
   3576 	if (all && In_Radio_Contact() && Contact_With_Whom()->As_Target() == target) {
   3577 		Transmit_Message(RADIO_OVER_OUT);
   3578 	}
   3579 }
   3580 
   3581 
   3582 /***********************************************************************************************
   3583  * TechnoClass::Kill_Cargo -- Destroys any cargo attached to this object.                      *
   3584  *                                                                                             *
   3585  *    This routine handles the destruction of any cargo this object may contain. Typical of    *
   3586  *    this would be when a transport helicopter gets destroyed.                                *
   3587  *                                                                                             *
   3588  * INPUT:   source   -- The source of the destruction of the cargo.                            *
   3589  *                                                                                             *
   3590  * OUTPUT:  none                                                                               *
   3591  *                                                                                             *
   3592  * WARNINGS:   none                                                                            *
   3593  *                                                                                             *
   3594  * HISTORY:                                                                                    *
   3595  *   07/29/1995 JLB : Created.                                                                 *
   3596  *=============================================================================================*/
   3597 void TechnoClass::Kill_Cargo(TechnoClass * source)
   3598 {
   3599 	while (Is_Something_Attached()) {
   3600 		FootClass * foot = Detach_Object();
   3601 		if (foot) {
   3602 			foot->Record_The_Kill(source);
   3603 			delete foot;
   3604 		}
   3605 	}
   3606 }
   3607 
   3608 
   3609 /***********************************************************************************************
   3610  * TechnoClass::Crew_Type -- Fetches the kind of crew this object contains.                    *
   3611  *                                                                                             *
   3612  *    This routine is called when generating survivors to this object. This routine returns    *
   3613  *    the type of survivor to generate.                                                        *
   3614  *                                                                                             *
   3615  * INPUT:   none                                                                               *
   3616  *                                                                                             *
   3617  * OUTPUT:  Returns the infantry type of a survivor.                                           *
   3618  *                                                                                             *
   3619  * WARNINGS:   This routine is designed to be called repeatedly. Once for each survivor to     *
   3620  *             generate.                                                                       *
   3621  *                                                                                             *
   3622  * HISTORY:                                                                                    *
   3623  *   07/29/1995 JLB : Created.                                                                 *
   3624  *=============================================================================================*/
   3625 InfantryType TechnoClass::Crew_Type(void) const
   3626 {
   3627 	InfantryType infantry = INFANTRY_E1;
   3628 	if (House->ActLike == HOUSE_NEUTRAL) {
   3629 		infantry = Random_Pick(INFANTRY_C1, INFANTRY_C9);
   3630 	} else {
   3631 		if (Techno_Type_Class()->Primary == WEAPON_NONE && Random_Pick(0, 6) == 1) {
   3632 			if (Random_Pick(0, 1) == 0) {
   3633 				infantry = INFANTRY_C1;
   3634 			} else {
   3635 				infantry = INFANTRY_C7;
   3636 			}
   3637 		}
   3638 	}
   3639 	return(infantry);
   3640 }
   3641 
   3642 
   3643 /***********************************************************************************************
   3644  * TechnoClass::Value -- Fetches the target value for this object.                             *
   3645  *                                                                                             *
   3646  *    This routine is used to fetch the target value for this object. The greater the value    *
   3647  *    returned, the better this object is as a target.                                         *
   3648  *                                                                                             *
   3649  * INPUT:   none                                                                               *
   3650  *                                                                                             *
   3651  * OUTPUT:  Returns with the target value for this object.                                     *
   3652  *                                                                                             *
   3653  * WARNINGS:   none                                                                            *
   3654  *                                                                                             *
   3655  * HISTORY:                                                                                    *
   3656  *   07/29/1995 JLB : Created.                                                                 *
   3657  *   08/16/1995 JLB : Adjusted for early mission lame-out.                                     *
   3658  *=============================================================================================*/
   3659 int TechnoClass::Value(void) const
   3660 {
   3661 	int value = 0;
   3662 
   3663 	/*
   3664 	**	In early missions, contents of transports are not figured
   3665 	**	into the total value. - 8/16/95
   3666 	*/
   3667 	if (BuildLevel > 8 || GameToPlay != GAME_NORMAL) {
   3668 		if (Is_Something_Attached()) {
   3669 			FootClass * object = Attached_Object();
   3670 
   3671 			while (object) {
   3672 				value += object->Value();
   3673 				object = (FootClass *)object->Next;
   3674 			}
   3675 		}
   3676 	}
   3677 	return Risk() + Techno_Type_Class()->Reward + value;
   3678 }
   3679 
   3680 
   3681 /***********************************************************************************************
   3682  * TechnoClass::Threat_Range -- Returns the range to scan based on threat control.             *
   3683  *                                                                                             *
   3684  *    This routine will return the range to scan based on the control value specified. The     *
   3685  *    value returned by this routine is typically used when scanning for enemies.              *
   3686  *                                                                                             *
   3687  * INPUT:   control  -- The range control parameter.                                           *
   3688  *                      0  = Use weapon range (zero is returned in this special case).         *
   3689  *                      -1 = Scan without range restrictions (-1 is returned in this case).    *
   3690  *                      1  = Scan up to twice weapon range.                                    *
   3691  *                                                                                             *
   3692  * OUTPUT:  Returns with a range (or special value) that can be used in the threat scan        *
   3693  *          process. If zero is returned, then always check threat against In_Range(). If      *
   3694  *          -1 is returned, then no range limitation restriction exists.                       *
   3695  *                                                                                             *
   3696  * WARNINGS:   none                                                                            *
   3697  *                                                                                             *
   3698  * HISTORY:                                                                                    *
   3699  *   07/29/1995 JLB : Created.                                                                 *
   3700  *=============================================================================================*/
   3701 int TechnoClass::Threat_Range(int control) const
   3702 {
   3703 	if (control == -1) return(-1);
   3704 	if (control == 0) return(0);
   3705 
   3706 	int range = MAX(Weapon_Range(0), Weapon_Range(1));
   3707 	range *= 2;
   3708 	range = Bound(range, 0x0000, 0x0A00);
   3709 	return(range);
   3710 }
   3711 
   3712 
   3713 /***********************************************************************************************
   3714  * TechnoClass::Base_Is_Attacked -- Handle panic response to base being attacked.              *
   3715  *                                                                                             *
   3716  *    This routine is called when the base is being attacked. It will pull units off of the    *
   3717  *    field and send them back to defend the base. This routine will make taking an enemy      *
   3718  *    base much more difficult.                                                                *
   3719  *                                                                                             *
   3720  * INPUT:   enemy -- Pointer to the enemy object that did the damage on the base.              *
   3721  *                                                                                             *
   3722  * OUTPUT:  none                                                                               *
   3723  *                                                                                             *
   3724  * WARNINGS:   This routine can drastically affect the game play. The computer will probably   *
   3725  *             call off its attacks as a result.                                               *
   3726  *                                                                                             *
   3727  * HISTORY:                                                                                    *
   3728  *   06/25/1995 JLB : Commented.                                                               *
   3729  *=============================================================================================*/
   3730 void TechnoClass::Base_Is_Attacked(TechnoClass const *enemy)
   3731 {
   3732 	FootClass *defender[6];
   3733 	int value[6];
   3734 	int count = 0;
   3735 	int weakest = 0;
   3736 	int desired = enemy->Risk() * 2;
   3737 	int risktotal = 0;
   3738 
   3739 	/*
   3740 	** Humans have to deal with their own base is attacked problems.
   3741 	*/
   3742 	if (!enemy || House->Is_Ally(enemy) || House->IsHuman) {
   3743 		return;
   3744 	}
   3745 
   3746 	/*
   3747 	**	Don't overreact if this building can defend itself.
   3748 	*/
   3749 	if (Techno_Type_Class()->Primary != WEAPON_NONE) return;
   3750 
   3751 	/*
   3752 	** If the enemy is not an infantry or a unit there is not much we can
   3753 	** do about it.
   3754 	*/
   3755 	if (enemy->What_Am_I() != RTTI_INFANTRY && enemy->What_Am_I() != RTTI_UNIT ) {
   3756 		return;
   3757 	}
   3758 
   3759 	/*
   3760 	**	If the enemy is a gunboat, then don't do anything.
   3761 	*/
   3762 	// This should allow helicopters to retaliate however. Hmmm.
   3763 	if (enemy->What_Am_I() == RTTI_UNIT && (*((UnitClass const *)enemy) == UNIT_GUNBOAT || *((UnitClass const *)enemy) == UNIT_HOVER)) {
   3764 		return;
   3765 	}
   3766 
   3767 	/*
   3768 	** If the threat has already been dealt with then we don't need to do
   3769 	** any work.  Check for that here.
   3770 	*/
   3771 	if (!((FootClass *)enemy)->BaseAttackTimer.Expired()) {
   3772 		return;
   3773 	}
   3774 
   3775 	/*
   3776 	** We will need units to defend our base.  We need to suspend teams until
   3777 	** the situation has been dealt with.
   3778 	*/
   3779 	TeamClass::Suspend_Teams(20);
   3780 
   3781 	/*
   3782 	** Loop through the infantry looking for those who are capable of going
   3783 	** on a rescue mission.
   3784 	*/
   3785 	int index;
   3786 	for (index = 0; index < Infantry.Count() && desired > 0; index++) {
   3787 	 	InfantryClass * infantry = Infantry.Ptr(index);
   3788 		if (infantry && infantry->Owner() == Owner()) {
   3789 
   3790 			/*
   3791 			**	Never recruite sticky guard units to defend a base.
   3792 			*/
   3793 			if (infantry->Mission == MISSION_STICKY || infantry->Mission == MISSION_SLEEP) continue;
   3794 
   3795 			/*
   3796 			** Find the amount of threat that this unit can apply to the
   3797 			** enemy.
   3798 			*/
   3799 			int threat = infantry->Rescue_Mission(enemy->As_Target());
   3800 
   3801 			/*
   3802 			** If it can't apply any threat then do just skip it and do not
   3803 			** add it to the list.
   3804 			*/
   3805 			if (!threat) {
   3806 				continue;
   3807 			}
   3808 
   3809 			/*
   3810 			** If the value returned is negative then this unit is already
   3811 			** assigned to fighting the enemy, so subtract its value from
   3812 			** the enemys desired value.
   3813 			*/
   3814 			if (threat < 0) {
   3815 				desired += threat;
   3816 				continue;
   3817 			}
   3818 
   3819 			if (count < 6) {
   3820 				defender[count] = (FootClass *)infantry;
   3821 				value[count] = threat;
   3822 				count++;
   3823 				continue;
   3824 			}
   3825 
   3826 			if (threat > weakest) {
   3827 				int newweakest = threat;
   3828 
   3829 				for (int lp = 0; lp < count; lp ++) {
   3830 					if (value[lp] == weakest) {
   3831 						value[lp] = threat;
   3832 						defender[lp] = (FootClass *) infantry;
   3833 						continue;
   3834 					}
   3835 					if (value[count] < newweakest) {
   3836 						newweakest = value[lp];
   3837 					}
   3838 				}
   3839 				weakest = newweakest;
   3840 			}
   3841 		}
   3842 	}
   3843 
   3844 	/*
   3845 	** Loop through the units looking for those who are capable of going
   3846 	** on a rescue mission.
   3847 	*/
   3848 	for (index = 0; index < Units.Count() && desired > 0; index++) {
   3849 	 	UnitClass * unit = Units.Ptr(index);
   3850 		if (unit && unit->Owner() == Owner()) {
   3851 
   3852 			/*
   3853 			**	Never recruite sticky guard units to defend a base.
   3854 			*/
   3855 			if (unit->Mission == MISSION_STICKY || unit->Mission == MISSION_SLEEP) continue;
   3856 
   3857 			/*
   3858 			** Find the amount of threat that this unit can apply to the
   3859 			** enemy.
   3860 			*/
   3861 			int threat = unit->Rescue_Mission(enemy->As_Target());
   3862 
   3863 			/*
   3864 			** If it can't apply any threat then do just skip it and do not
   3865 			** add it to the list.
   3866 			*/
   3867 			if (!threat) {
   3868 				continue;
   3869 			}
   3870 
   3871 			/*
   3872 			** If the value returned is negative then this unit is already
   3873 			** assigned to fighting the enemy, so subtract its value from
   3874 			** the enemys desired value.
   3875 			*/
   3876 			if (threat < 0) {
   3877 				desired += threat;
   3878 				continue;
   3879 			}
   3880 
   3881 			if (count < 6) {
   3882 				defender[count] = (FootClass *)unit;
   3883 				value[count] = threat;
   3884 				count++;
   3885 				continue;
   3886 			}
   3887 			if (threat > weakest) {
   3888 				int newweakest = threat;
   3889 
   3890 				for (int lp = 0; lp < count; lp ++) {
   3891 					if (value[lp] == weakest) {
   3892 						value[lp] = threat;
   3893 						defender[lp] = (FootClass *) unit;
   3894 						continue;
   3895 					}
   3896 					if (value[count] < newweakest) {
   3897 						newweakest = value[lp];
   3898 					}
   3899 				}
   3900 				weakest = newweakest;
   3901 			}
   3902 		}
   3903 	}
   3904 
   3905 	if (desired > 0) {
   3906 
   3907 		/*
   3908 		** Sort the defenders by value, this doesn't take very long and will
   3909 		** help the closest defenders to respond.
   3910 		*/
   3911 		int lp;
   3912 		for (lp = 0; lp < count - 1; lp ++) {
   3913 			for (int lp2 = lp + 1; lp2 < count; lp2++) {
   3914 				if (value[lp] < value[lp2]) {
   3915 
   3916 					value[lp] 	^= value[lp2];
   3917 					value[lp2]	^= value[lp];
   3918 					value[lp]	^= value[lp2];
   3919 
   3920 					FootClass *temp;
   3921 					temp				= defender[lp];
   3922 					defender[lp]	= defender[lp2];
   3923 					defender[lp2]  = temp;
   3924 				}
   3925 			}
   3926 		}
   3927 
   3928 		for (lp = 0; lp < count; lp ++) {
   3929 			defender[lp]->Assign_Mission(MISSION_RESCUE);
   3930 			defender[lp]->Assign_Target(enemy->As_Target());
   3931 			risktotal += defender[lp]->Risk();
   3932 			if (risktotal > desired) {
   3933 				break;
   3934 			}
   3935 		}
   3936 	}
   3937 
   3938 	if (risktotal > desired) {
   3939 		((FootClass *)enemy)->BaseAttackTimer.Set(15 * 15);
   3940 	}
   3941 }
   3942 
   3943 
   3944 /***********************************************************************************************
   3945  * TechnoClass::Get_Ownable -- Fetches the ownable bits for this object.                       *
   3946  *                                                                                             *
   3947  *    This routine will return the ownable bits for this object. The ownable bits represent    *
   3948  *    the houses that are allowed to own this object.                                          *
   3949  *                                                                                             *
   3950  * INPUT:   none                                                                               *
   3951  *                                                                                             *
   3952  * OUTPUT:  Returns the ownable bits for this object.                                          *
   3953  *                                                                                             *
   3954  * WARNINGS:   none                                                                            *
   3955  *                                                                                             *
   3956  * HISTORY:                                                                                    *
   3957  *   07/29/1995 JLB : Created.                                                                 *
   3958  *=============================================================================================*/
   3959 unsigned char TechnoClass::Get_Ownable(void) const
   3960 {
   3961 	return ((TechnoTypeClass const &)Class_Of()).Ownable;
   3962 }
   3963 
   3964 
   3965 /***********************************************************************************************
   3966  * TechnoClass::Is_Techno -- Confirms that this is a TechnoClass object.                       *
   3967  *                                                                                             *
   3968  *    This routine is called to confirm if this object is derived from the TechnoClass         *
   3969  *    object. At this level, the return value will always be true.                             *
   3970  *                                                                                             *
   3971  * INPUT:   none                                                                               *
   3972  *                                                                                             *
   3973  * OUTPUT:  Is this object a TechnoClass or derived from it?                                   *
   3974  *                                                                                             *
   3975  * WARNINGS:   none                                                                            *
   3976  *                                                                                             *
   3977  * HISTORY:                                                                                    *
   3978  *   07/29/1995 JLB : Created.                                                                 *
   3979  *=============================================================================================*/
   3980 bool TechnoClass::Is_Techno(void) const
   3981 {
   3982 	return(true);
   3983 }
   3984 
   3985 
   3986 /***********************************************************************************************
   3987  * TechnoClass::Risk -- Fetches the risk associated with this object.                          *
   3988  *                                                                                             *
   3989  *    This routine is called when the risk value for this object needs to be determined.       *
   3990  *                                                                                             *
   3991  * INPUT:   none                                                                               *
   3992  *                                                                                             *
   3993  * OUTPUT:  Returns with the risk value for this object.                                       *
   3994  *                                                                                             *
   3995  * WARNINGS:   none                                                                            *
   3996  *                                                                                             *
   3997  * HISTORY:                                                                                    *
   3998  *   07/29/1995 JLB : Created.                                                                 *
   3999  *=============================================================================================*/
   4000 int TechnoClass::Risk(void) const
   4001 {
   4002 	return(Techno_Type_Class()->Risk);
   4003 }
   4004 
   4005 
   4006 /***********************************************************************************************
   4007  * TechnoClass::Tiberium_Load -- Fetches the current tiberium load percentage.                 *
   4008  *                                                                                             *
   4009  *    This routine will return the current Tiberium load (expressed as a fixed point fraction) *
   4010  *    that this object currently contains. Typical implementor of this function would be       *
   4011  *    the harvester. Any object that can return a non-zero value should derive from this       *
   4012  *    function in order to return the appropriate value.                                       *
   4013  *                                                                                             *
   4014  * INPUT:   none                                                                               *
   4015  *                                                                                             *
   4016  * OUTPUT:  Returns with the current Tiberium load expressed as a fixed point number.          *
   4017  *          0x0000   = empty                                                                   *
   4018  *          0x0080   = half full                                                               *
   4019  *          0x0100   = full                                                                    *
   4020  *                                                                                             *
   4021  * WARNINGS:   none                                                                            *
   4022  *                                                                                             *
   4023  * HISTORY:                                                                                    *
   4024  *   07/29/1995 JLB : Created.                                                                 *
   4025  *=============================================================================================*/
   4026 int TechnoClass::Tiberium_Load(void) const
   4027 {
   4028 	return(0x0000);
   4029 }
   4030 
   4031 
   4032 /***********************************************************************************************
   4033  * TechnoClass::Desired_Load_Dir -- Fetches loading parameters for this object.                *
   4034  *                                                                                             *
   4035  *    This routine is called when an object desires to load up on this object. The object      *
   4036  *    desiring to load is specified. The cell that the loading object should move to is        *
   4037  *    determined. The direction that this object should face is also calculated. This routine  *
   4038  *    will be overridden by those objects that can actually load up passengers.                *
   4039  *                                                                                             *
   4040  * INPUT:   object   -- The object that is desiring to load up.                                *
   4041  *                                                                                             *
   4042  *          moveto   -- Reference to the cell that the loading object should move to before    *
   4043  *                      the final load process occurs (this value will be filled in).          *
   4044  *                                                                                             *
   4045  * OUTPUT:  Returns with the direction that the transport object should face.                  *
   4046  *                                                                                             *
   4047  * WARNINGS:   none                                                                            *
   4048  *                                                                                             *
   4049  * HISTORY:                                                                                    *
   4050  *   07/29/1995 JLB : Created.                                                                 *
   4051  *=============================================================================================*/
   4052 DirType TechnoClass::Desired_Load_Dir(ObjectClass * , CELL & moveto) const
   4053 {
   4054 	moveto = 0;
   4055 	return(DIR_N);
   4056 }
   4057 
   4058 
   4059 /***********************************************************************************************
   4060  * TechnoClass::Pip_Count -- Fetches the number of pips to display on this object.             *
   4061  *                                                                                             *
   4062  *    This routine will return the number of pips to display on this object when the object    *
   4063  *    is selected. The default condition is to return no pips at all. This routine is          *
   4064  *    derived for those objects that can have pips.                                            *
   4065  *                                                                                             *
   4066  * INPUT:   none                                                                               *
   4067  *                                                                                             *
   4068  * OUTPUT:  Returns with the number of pips to display on this object when selected.           *
   4069  *                                                                                             *
   4070  * WARNINGS:   none                                                                            *
   4071  *                                                                                             *
   4072  * HISTORY:                                                                                    *
   4073  *   07/29/1995 JLB : Created.                                                                 *
   4074  *=============================================================================================*/
   4075 int TechnoClass::Pip_Count(void) const
   4076 {
   4077 	return(0);
   4078 }
   4079 
   4080 
   4081 /***********************************************************************************************
   4082  * TechnoClass::Fire_Direction -- Fetches the direction projectile fire will take.             *
   4083  *                                                                                             *
   4084  *    This routine will fetch the direction that a fired projectile will take. This is         *
   4085  *    usually the facing of the object's weapon. This routine will be derived for the objects  *
   4086  *    that have their weapon barrel facing a different direction than the body.                *
   4087  *                                                                                             *
   4088  * INPUT:   none                                                                               *
   4089  *                                                                                             *
   4090  * OUTPUT:  Returns with the direction a fired projectile will take.                           *
   4091  *                                                                                             *
   4092  * WARNINGS:   none                                                                            *
   4093  *                                                                                             *
   4094  * HISTORY:                                                                                    *
   4095  *   07/29/1995 JLB : Created.                                                                 *
   4096  *=============================================================================================*/
   4097 DirType TechnoClass::Fire_Direction(void) const
   4098 {
   4099 	return(PrimaryFacing.Current());
   4100 }
   4101 
   4102 
   4103 /***********************************************************************************************
   4104  * TechnoClass::Response_Select -- Handles the voice response when selected.                   *
   4105  *                                                                                             *
   4106  *    This routine is called when a voice reponse to a select action is desired. This routine  *
   4107  *    should be overridden for any object that actually has a voice response.                  *
   4108  *                                                                                             *
   4109  * INPUT:   none                                                                               *
   4110  *                                                                                             *
   4111  * OUTPUT:  none                                                                               *
   4112  *                                                                                             *
   4113  * WARNINGS:   This routine can generate an audio response.                                    *
   4114  *                                                                                             *
   4115  * HISTORY:                                                                                    *
   4116  *   07/29/1995 JLB : Created.                                                                 *
   4117  *=============================================================================================*/
   4118 void TechnoClass::Response_Select(void)
   4119 {
   4120 }
   4121 
   4122 
   4123 /***********************************************************************************************
   4124  * TechnoClass::Response_Move -- Handles the voice repsonse to a movement request.             *
   4125  *                                                                                             *
   4126  *    This routine is called when a voice response to a movement order is desired. This        *
   4127  *    routine should be overridden for any object that actually has a voice response.          *
   4128  *                                                                                             *
   4129  * INPUT:   none                                                                               *
   4130  *                                                                                             *
   4131  * OUTPUT:  none                                                                               *
   4132  *                                                                                             *
   4133  * WARNINGS:   This can generate an audio repsonse.                                            *
   4134  *                                                                                             *
   4135  * HISTORY:                                                                                    *
   4136  *   07/29/1995 JLB : Created.                                                                 *
   4137  *=============================================================================================*/
   4138 void TechnoClass::Response_Move(void)
   4139 {
   4140 }
   4141 
   4142 
   4143 /***********************************************************************************************
   4144  * TechnoClass::Response_Attack -- Handles the voice response when given attack order.         *
   4145  *                                                                                             *
   4146  *    This routine is called when a voice response to an attack order is desired. This routine *
   4147  *    should be overridden for any object that actually have a voice response.                 *
   4148  *                                                                                             *
   4149  * INPUT:   none                                                                               *
   4150  *                                                                                             *
   4151  * OUTPUT:  none                                                                               *
   4152  *                                                                                             *
   4153  * WARNINGS:   This can generate an audio response.                                            *
   4154  *                                                                                             *
   4155  * HISTORY:                                                                                    *
   4156  *   07/29/1995 JLB : Created.                                                                 *
   4157  *=============================================================================================*/
   4158 void TechnoClass::Response_Attack(void)
   4159 {
   4160 }
   4161 
   4162 
   4163 /***********************************************************************************************
   4164  * TechnoClass::Target_Something_Nearby -- Handles finding and assigning a nearby target.      *
   4165  *                                                                                             *
   4166  *    This routine will search for a nearby target and assign it to this object's TarCom.      *
   4167  *    The method to use when scanning for a target is controlled by the parameter passed.      *
   4168  *                                                                                             *
   4169  * INPUT:   threat   -- The threat control parameter used to control the range searched. The   *
   4170  *                      only values recognized are THREAT_RANGE and THREAT_AREA.               *
   4171  *                                                                                             *
   4172  * OUTPUT:  Was a suitable target aquired and assigned to the TarCom?                          *
   4173  *                                                                                             *
   4174  * WARNINGS:   none                                                                            *
   4175  *                                                                                             *
   4176  * HISTORY:                                                                                    *
   4177  *   07/29/1995 JLB : Created.                                                                 *
   4178  *=============================================================================================*/
   4179 bool TechnoClass::Target_Something_Nearby(ThreatType threat)
   4180 {
   4181 	threat = threat & (THREAT_RANGE|THREAT_AREA);
   4182 
   4183 	/*
   4184 	**	Determine that if there is an existing target it is still legal
   4185 	**	and within range.
   4186 	*/
   4187 	if (Target_Legal(TarCom)) {
   4188 		if ((threat & THREAT_RANGE) && !In_Range(TarCom)) {
   4189 			Assign_Target(TARGET_NONE);
   4190 		}
   4191 	}
   4192 
   4193 	/*
   4194 	**	If there is no target, then try to find one and assign it as
   4195 	**	the target for this unit.
   4196 	*/
   4197 	if (!Target_Legal(TarCom)) {
   4198 		Assign_Target(Greatest_Threat(threat));
   4199 	}
   4200 
   4201 	/*
   4202 	**	Return with answer to question: Does this unit have a target?
   4203 	*/
   4204 	return(Target_Legal(TarCom));
   4205 }
   4206 
   4207 
   4208 /***********************************************************************************************
   4209  * TechnoClass::Exit_Object -- Causes specified object to leave this object.                   *
   4210  *                                                                                             *
   4211  *    This routine is called when there is an attached object that should detach and leave     *
   4212  *    this object. Typical of this would be the refinery and APC.                              *
   4213  *                                                                                             *
   4214  * INPUT:   object   -- The object that is trying to leave this object.                        *
   4215  *                                                                                             *
   4216  * OUTPUT:  Was the object successfully launched from this object? Failure might indicate that *
   4217  *          there is insufficient room to detach the specified object.                         *
   4218  *                                                                                             *
   4219  * WARNINGS:   none                                                                            *
   4220  *                                                                                             *
   4221  * HISTORY:                                                                                    *
   4222  *   07/24/1995 JLB : Created.                                                                 *
   4223  *=============================================================================================*/
   4224 int TechnoClass::Exit_Object(TechnoClass *)
   4225 {
   4226 	return(0);
   4227 }
   4228 
   4229 
   4230 /***********************************************************************************************
   4231  * TechnoClass::Random_Animate -- Performs some idle animation for the object.                 *
   4232  *                                                                                             *
   4233  *    This is a maintenance routine that is called when the object might want to check to see  *
   4234  *    if it should go into some idle animation. Infantry are a good example of objects that    *
   4235  *    perform idle animations. This routine must be overridden by the derived object types     *
   4236  *    in order to give it some functionality.                                                  *
   4237  *                                                                                             *
   4238  * INPUT:   none                                                                               *
   4239  *                                                                                             *
   4240  * OUTPUT:  none                                                                               *
   4241  *                                                                                             *
   4242  * WARNINGS:   none                                                                            *
   4243  *                                                                                             *
   4244  * HISTORY:                                                                                    *
   4245  *   07/24/1995 JLB : Created.                                                                 *
   4246  *=============================================================================================*/
   4247 void TechnoClass::Random_Animate(void)
   4248 {
   4249 }
   4250 
   4251 
   4252 /***********************************************************************************************
   4253  * TechnoClass::Assign_Destination -- Assigns movement destination to the object.              *
   4254  *                                                                                             *
   4255  *    This routine is called when the object needs to have a new movement destination          *
   4256  *    assigned. This routine must be overridden since at this level, movement is not allowed.  *
   4257  *                                                                                             *
   4258  * INPUT:   destination -- The destination to assign to this object.                           *
   4259  *                                                                                             *
   4260  * OUTPUT:  none                                                                               *
   4261  *                                                                                             *
   4262  * WARNINGS:   none                                                                            *
   4263  *                                                                                             *
   4264  * HISTORY:                                                                                    *
   4265  *   07/24/1995 JLB : Created.                                                                 *
   4266  *=============================================================================================*/
   4267 void TechnoClass::Assign_Destination(TARGET )
   4268 {
   4269 }
   4270 
   4271 
   4272 /***********************************************************************************************
   4273  * TechnoClass::Scatter -- Causes the object to scatter to an adjacent cell.                   *
   4274  *                                                                                             *
   4275  *    This routine is called when the object needs to get out of the way. This might be as a   *
   4276  *    result of combat or findpath reasons.                                                    *
   4277  *                                                                                             *
   4278  * INPUT:   coord    -- The source of the reason to scatter. The object should try to run      *
   4279  *                      away from this coordinate.                                             *
   4280  *                                                                                             *
   4281  *          forced   -- Is the scatter a forced scatter? If false, then this is merely a       *
   4282  *                      request that scattering might be a good idea.                          *
   4283  *                                                                                             *
   4284  * OUTPUT:  none                                                                               *
   4285  *                                                                                             *
   4286  * WARNINGS:   none                                                                            *
   4287  *                                                                                             *
   4288  * HISTORY:                                                                                    *
   4289  *   07/24/1995 JLB : Created.                                                                 *
   4290  *=============================================================================================*/
   4291 void TechnoClass::Scatter(COORDINATE , bool , bool )
   4292 {
   4293 }
   4294 
   4295 
   4296 /***********************************************************************************************
   4297  * TechnoClass::Enter_Idle_Mode -- Object enters its default idle condition.                   *
   4298  *                                                                                             *
   4299  *    This routine is called when the object should intelligently revert to an idle state.     *
   4300  *    Typically this routine is called after some mission has completed. This routine must     *
   4301  *    be overridden by the various object types. It is located at this level merely to provide *
   4302  *    a virtual function entry point.                                                          *
   4303  *                                                                                             *
   4304  * INPUT:   initial  -- Is this called when the unit just leaves a factory or is initially     *
   4305  *                      or is initially placed on the map?                                     *
   4306  *                                                                                             *
   4307  * OUTPUT:  none                                                                               *
   4308  *                                                                                             *
   4309  * WARNINGS:   none                                                                            *
   4310  *                                                                                             *
   4311  * HISTORY:                                                                                    *
   4312  *   07/24/1995 JLB : Created.                                                                 *
   4313  *=============================================================================================*/
   4314 void TechnoClass::Enter_Idle_Mode(bool )
   4315 {
   4316 }
   4317 
   4318 
   4319 /***********************************************************************************************
   4320  * TechnoClass::Draw_Pips -- Draws the transport pips and other techno graphics.               *
   4321  *                                                                                             *
   4322  *    This routine is used to render the small transportation pip (occupant feedback graphic)  *
   4323  *    used for transporter object. It will also display if the techno object is "primary"      *
   4324  *    if necessary.                                                                            *
   4325  *                                                                                             *
   4326  * INPUT:   x,y   -- The pixel coordinate for the center of the first pip. Subsiquent pips     *
   4327  *                   are drawn rightward.                                                      *
   4328  *                                                                                             *
   4329  *          window-- The window that pip clipping is relative to.                              *
   4330  *                                                                                             *
   4331  * OUTPUT:  none                                                                               *
   4332  *                                                                                             *
   4333  * WARNINGS:   none                                                                            *
   4334  *                                                                                             *
   4335  * HISTORY:                                                                                    *
   4336  *   08/08/1995 JLB : Created.                                                                 *
   4337  *=============================================================================================*/
   4338 void TechnoClass::Draw_Pips(int x, int y, WindowNumberType window)
   4339 {
   4340 	/*
   4341 	**	Transporter type objects have a different graphic representation for the pips. The
   4342 	**	pip color represents the type of occupant.
   4343 	*/
   4344 	if (Techno_Type_Class()->IsTransporter) {
   4345 		ObjectClass const * object = Attached_Object();
   4346 
   4347 		for (int index = 0; index < Class_Of().Max_Pips(); index++) {
   4348 			PipEnum pip = PIP_EMPTY;
   4349 
   4350 			if (object) {
   4351 				pip = PIP_FULL;
   4352 				if (object->What_Am_I() == RTTI_INFANTRY) {
   4353 					if (*((InfantryClass *)object) == INFANTRY_RAMBO) {
   4354 						pip = PIP_COMMANDO;
   4355 					}
   4356 					if (*((InfantryClass *)object) == INFANTRY_E7) {
   4357 						pip = PIP_ENGINEER;
   4358 					}
   4359 					if (((InfantryClass *)object)->Class->IsCivilian) {
   4360 						pip = PIP_CIVILIAN;
   4361 					}
   4362 				}
   4363 				object = object->Next;
   4364 			}
   4365 			CC_Draw_Pip(this, Class_Of().PipShapes, pip, x+index*3, y, window, SHAPE_CENTER|SHAPE_WIN_REL);
   4366 		}
   4367 
   4368 	} else {
   4369 
   4370 		/*
   4371 		**	Display number of how many attached objects there are. This is also used
   4372 		**	to display the fullness rating for a harvester.
   4373 		*/
   4374 		int pips = Pip_Count();
   4375 		for (int index = 0; index < Class_Of().Max_Pips(); index++) {
   4376 			CC_Draw_Pip(this, Class_Of().PipShapes, (index < pips) ? PIP_FULL : PIP_EMPTY, x+index*3, y, window, SHAPE_CENTER|SHAPE_WIN_REL);
   4377 		}
   4378 	}
   4379 
   4380 	/*
   4381 	**	Display whether this unit is a leader unit or not.
   4382 	*/
   4383 	if (IsLeader && (window != WINDOW_VIRTUAL)) {
   4384 		CC_Draw_Pip(this, Class_Of().PipShapes, PIP_PRIMARY, x-2, y-3, window, /*SHAPE_CENTER|*/SHAPE_WIN_REL);
   4385 	}
   4386 }
   4387 
   4388 
   4389 /***********************************************************************************************
   4390  * TechnoClass::Find_Docking_Bay -- Searches for a close docking bay.                          *
   4391  *                                                                                             *
   4392  *    This routine will be used to find a building that can serve as a docking bay. The        *
   4393  *    closest building that qualifies will be returned. If no building could be found then     *
   4394  *    return with NULL.                                                                        *
   4395  *                                                                                             *
   4396  * INPUT:   b  -- The structure type to look for.                                              *
   4397  *                                                                                             *
   4398  *          friendly -- Allow searching for allied buildings as well.                          *
   4399  *                                                                                             *
   4400  * OUTPUT:  Returns with a pointer to the building that can serve as the best docking bay.     *
   4401  *                                                                                             *
   4402  * WARNINGS:   This routine might return NULL even if there are buildings of the specified     *
   4403  *             type available. This is the case when the building(s) are currently busy and    *
   4404  *             cannot serve as a docking bay at the moment.                                    *
   4405  *                                                                                             *
   4406  * HISTORY:                                                                                    *
   4407  *   07/18/1995 JLB : Created.                                                                 *
   4408  *   08/13/1995 JLB : Recognizes the "IsLeader" method of building preference.                 *
   4409  *=============================================================================================*/
   4410 BuildingClass * TechnoClass::Find_Docking_Bay(StructType b, bool friendly) const
   4411 {
   4412 	BuildingClass * best = 0;
   4413 
   4414 	/*
   4415 	**	First check to see if there are ANY buildings of the specified
   4416 	**	type in thi house's inventory. If not, then don't bother to scan
   4417 	**	for one.
   4418 	*/
   4419 	if (House->BScan & (1L << b)) {
   4420 		int bestval = -1;
   4421 
   4422 		/*
   4423 		**	Loop through all the buildings and find the one that matches the specification
   4424 		**	and is willing to dock with this object.
   4425 		*/
   4426 		for (int index = 0; index < Buildings.Count(); index++) {
   4427 			BuildingClass * building = Buildings.Ptr(index);
   4428 
   4429 			/*
   4430 			**	Check to see if the building qualifies (preliminary scan).
   4431 			*/
   4432 			if (building &&
   4433 				(friendly ? building->House->Is_Ally(this) : building->House == House) &&
   4434 				!building->IsInLimbo &&
   4435 				*building == b &&
   4436 				((TechnoClass *)this)->Transmit_Message(RADIO_CAN_LOAD, building) == RADIO_ROGER) {
   4437 
   4438 				/*
   4439 				**	If the building qualifies and this building is better than the
   4440 				**	last qualifying building (as rated by distance), then record
   4441 				**	this building and keep scanning.
   4442 				*/
   4443 				if (bestval == -1 || Distance(building) < bestval || building->IsLeader) {
   4444 					best = building;
   4445 					bestval = Distance(building);
   4446 					if (building->IsLeader) break;
   4447 				}
   4448 			}
   4449 		}
   4450 	}
   4451 	return(best);
   4452 }
   4453 
   4454 
   4455 /***********************************************************************************************
   4456  * TechnoClass::Find_Exit_Cell -- Finds an appropriate exit cell for this object.              *
   4457  *                                                                                             *
   4458  *    This routine is called when an object would like to exit from this (presumed) transport. *
   4459  *    A suitable cell should be returned by this routine. The specified object will probably   *
   4460  *    be unloaded at that cell.                                                                *
   4461  *                                                                                             *
   4462  * INPUT:   techno   -- Pointer to the object that would like to unload. This is used to       *
   4463  *                      determine suitability for placement.                                   *
   4464  *                                                                                             *
   4465  * OUTPUT:  Returns with the cell that is recommended for object exit.                         *
   4466  *                                                                                             *
   4467  * WARNINGS:   none                                                                            *
   4468  *                                                                                             *
   4469  * HISTORY:                                                                                    *
   4470  *   08/12/1995 JLB : Created.                                                                 *
   4471  *=============================================================================================*/
   4472 CELL TechnoClass::Find_Exit_Cell(TechnoClass const * ) const
   4473 {
   4474 	return(Coord_Cell(Docking_Coord()));
   4475 }
   4476 
   4477 
   4478 /***********************************************************************************************
   4479  * TechnoClass::Refund_Amount -- Returns with the money to refund if this object is sold.      *
   4480  *                                                                                             *
   4481  *    This routine is used by the selling back mechanism in order to credit the owning house   *
   4482  *    with some refund credits. The value returned is the credits to refund to the owner.      *
   4483  *                                                                                             *
   4484  * INPUT:   none                                                                               *
   4485  *                                                                                             *
   4486  * OUTPUT:  Returns with the credits to refund to the owner.                                   *
   4487  *                                                                                             *
   4488  * WARNINGS:   none                                                                            *
   4489  *                                                                                             *
   4490  * HISTORY:                                                                                    *
   4491  *   08/13/1995 JLB : Created.                                                                 *
   4492  *=============================================================================================*/
   4493 int TechnoClass::Refund_Amount(void) const
   4494 {
   4495 	int cost = Techno_Type_Class()->Raw_Cost() * House->CostBias;
   4496 
   4497 	/*
   4498 	**	If the object is carrying Tiberium directly (i.e., the harvester), then
   4499 	**	account for the credits of the load.
   4500 	*/
   4501 	cost += Fixed_To_Cardinal(UnitTypeClass::FULL_LOAD_CREDITS, Tiberium_Load())/2;
   4502 
   4503 	if (House->IsHuman) {
   4504 		cost /= 2;
   4505 	}
   4506 	return(cost);
   4507 }
   4508 
   4509 
   4510 
   4511 
   4512 
   4513 
   4514 
   4515 
   4516 /*
   4517 **  Additions to TechnoClass to track discovery per-player. ST - 3/6/2019 11:18AM
   4518 ** 
   4519 ** 
   4520 ** 
   4521 ** 
   4522 ** 
   4523 */
   4524 
   4525 
   4526 /***********************************************************************************************
   4527  * TechnoClass::Is_Discovered_By_Player -- Has this object been disovered by the given player  *
   4528  *                                                                                             *
   4529  * INPUT:   Player pointer                                                                     *
   4530  *                                                                                             *
   4531  * OUTPUT:  True if discovered by that player                                                  *
   4532  *                                                                                             *
   4533  * WARNINGS:   none                                                                            *
   4534  *                                                                                             *
   4535  * HISTORY:                                                                                    *
   4536  *   3/6/2019 11:20AM - ST                                                                     *
   4537  *=============================================================================================*/
   4538 bool TechnoClass::Is_Discovered_By_Player(HouseClass *player) const
   4539 {
   4540 	if (player && player->Class) {
   4541 		int shift = (int) player->Class->House;
   4542 		return (IsDiscoveredByPlayerMask & (1 << shift)) ? true : false;
   4543 	} else {
   4544 		int shift = (int) PlayerPtr->Class->House;
   4545 		return (IsDiscoveredByPlayerMask & (1 << shift)) ? true : false;
   4546 	}	
   4547 	return false;
   4548 }
   4549 	
   4550 
   4551 /***********************************************************************************************
   4552  * TechnoClass::Set_Discovered_By_Player -- Mark that this object been disovered by the player *
   4553  *                                                                                             *
   4554  * INPUT:   Player pointer                                                                     *
   4555  *                                                                                             *
   4556  * OUTPUT:                                                                                     *
   4557  *                                                                                             *
   4558  * WARNINGS:   none                                                                            *
   4559  *                                                                                             *
   4560  * HISTORY:                                                                                    *
   4561  *   3/6/2019 11:23AM - ST                                                                     *
   4562  *=============================================================================================*/
   4563 void TechnoClass::Set_Discovered_By_Player(HouseClass *player)
   4564 {
   4565 	int shift = 0;
   4566 	if (player && player->Class) {
   4567 		shift = (int) player->Class->House;
   4568 	} else {
   4569 		shift = (int)PlayerPtr->Class->House;
   4570 	}	
   4571 	IsDiscoveredByPlayerMask |= (1 << shift);
   4572 
   4573 	if (GameToPlay == GAME_NORMAL && player == PlayerPtr) {
   4574 		IsDiscoveredByPlayer = true;
   4575 	}
   4576 }
   4577 
   4578 
   4579 
   4580 /***********************************************************************************************
   4581  * TechnoClass::Clear_Discovered_By_Players -- Clear player discovery flags                    *
   4582  *                                                                                             *
   4583  * INPUT:   Player pointer                                                                     *
   4584  *                                                                                             *
   4585  * OUTPUT:                                                                                     *
   4586  *                                                                                             *
   4587  * WARNINGS:   none                                                                            *
   4588  *                                                                                             *
   4589  * HISTORY:                                                                                    *
   4590  *   4/27/2020 - SKY                                                                           *
   4591  *=============================================================================================*/
   4592 void TechnoClass::Clear_Discovered_By_Players()
   4593 {
   4594 	IsDiscoveredByPlayerMask = 0;
   4595 	IsDiscoveredByPlayer = false;
   4596 }
   4597 
   4598 
   4599 
   4600 /***********************************************************************************************
   4601  * TechnoClass::Is_Owned_By_Player -- Is this object owned by the active human player          *
   4602  *                                                                                             *
   4603  * INPUT:   Player pointer                                                                     *
   4604  *                                                                                             *
   4605  * OUTPUT:                                                                                     *
   4606  *                                                                                             *
   4607  * WARNINGS:   none                                                                            *
   4608  *                                                                                             *
   4609  * HISTORY:                                                                                    *
   4610  *   3/13/2019 5:00PM - ST                                                                     *
   4611  *=============================================================================================*/
   4612 bool TechnoClass::Is_Owned_By_Player(HouseClass *player) const
   4613 {
   4614 	if (player == NULL) {
   4615 		return (House == PlayerPtr) ? true : false;
   4616 	}
   4617 	return (House == player) ? true : false;
   4618 }			  
   4619 
   4620 
   4621 bool TechnoClass::Is_Cloaked(HousesType house) const
   4622 {
   4623 	return !House->Is_Ally(house) && (Cloak == CLOAKED);
   4624 }
   4625 
   4626 
   4627 bool TechnoClass::Is_Cloaked(HouseClass const * house) const
   4628 {
   4629 	return !House->Is_Ally(house) && (Cloak == CLOAKED);
   4630 }
   4631 
   4632 
   4633 bool TechnoClass::Is_Cloaked(ObjectClass const * object) const
   4634 {
   4635 	return !House->Is_Ally(object) && (Cloak == CLOAKED);
   4636 }
   4637 
   4638 
   4639 
   4640 
   4641 
   4642 
   4643 
   4644 
   4645 #ifdef USE_RA_AI
   4646 
   4647  /*
   4648  ** Additions for AI, from Red Alert. ST - 7/16/2019 11:36AM
   4649  */
   4650 
   4651 
   4652 /***********************************************************************************************
   4653  * TechnoClass::Anti_Air -- Determines the anti-aircraft strength of the object.               *
   4654  *                                                                                             *
   4655  *    This routine will calculate and return the anti-aircraft strength of this object.        *
   4656  *    Typical users of this strength value is the base defense expert system AI.               *
   4657  *                                                                                             *
   4658  * INPUT:   none                                                                               *
   4659  *                                                                                             *
   4660  * OUTPUT:  Returns with the anti-aircraft defense value of this object. The value returned    *
   4661  *          is an abstract number to be used for relative comparisons only.                    *
   4662  *                                                                                             *
   4663  * WARNINGS:   none                                                                            *
   4664  *                                                                                             *
   4665  * HISTORY:                                                                                    *
   4666  *   10/02/1995 JLB : Created.                                                                 *
   4667  *=============================================================================================*/
   4668 int TechnoClass::Anti_Air(void) const
   4669 {
   4670 	//assert(IsActive);
   4671 
   4672 	if (Is_Weapon_Equipped()) {
   4673 		WeaponType weapon_type = Techno_Type_Class()->Primary;
   4674 		WeaponTypeClass const * weapon = &Weapons[weapon_type];
   4675 		//BulletTypeClass const * bullet = weapon->Bullet;
   4676 		const BulletTypeClass &bullet = BulletTypeClass::As_Reference(Weapons[weapon_type].Fires);
   4677 		//WarheadTypeClass const * warhead = weapon->WarheadPtr;
   4678 		WarheadTypeClass const * warhead = &Warheads[bullet.Warhead];
   4679 
   4680 
   4681 		//if (bullet->IsAntiAircraft) {
   4682 		if (weapon_type != WEAPON_NONE && BulletTypeClass::As_Reference(Weapons[weapon_type].Fires).IsAntiAircraft) {
   4683 			int value = ((weapon->Attack * warhead->Modifier[ARMOR_ALUMINUM]) * weapon->Range) / weapon->ROF;
   4684 
   4685 			//if (Techno_Type_Class()->Is_Two_Shooter()) {
   4686 			//	value *= 2;
   4687 			//}
   4688 			return(value/50);
   4689 		}
   4690 	}
   4691 	return(0);
   4692 }
   4693 
   4694 
   4695 /***********************************************************************************************
   4696  * TechnoClass::Anti_Armor -- Determines the anti-armor strength of the object.                *
   4697  *                                                                                             *
   4698  *    This routine is used to examine and calculate the anti-armor strength of this object.    *
   4699  *    Typical user user of this would be the expert system base defense AI.                    *
   4700  *                                                                                             *
   4701  * INPUT:   none                                                                               *
   4702  *                                                                                             *
   4703  * OUTPUT:  Returns with the relative anti-armor combat value for this object. The value       *
   4704  *          is abstract and is only to be used in relative comparisons.                        *
   4705  *                                                                                             *
   4706  * WARNINGS:   none                                                                            *
   4707  *                                                                                             *
   4708  * HISTORY:                                                                                    *
   4709  *   10/02/1995 JLB : Created.                                                                 *
   4710  *=============================================================================================*/
   4711 int TechnoClass::Anti_Armor(void) const
   4712 {
   4713 	//assert(IsActive);
   4714 
   4715 	if (Is_Weapon_Equipped()) {
   4716 
   4717 		WeaponType weapon_type = Techno_Type_Class()->Primary;
   4718 		//if (!Techno_Type_Class()->PrimaryWeapon->Bullet->IsAntiGround) return(0);
   4719 		if (weapon_type != WEAPON_NONE && BulletTypeClass::As_Reference(Weapons[weapon_type].Fires).IsAntiAircraft) {
   4720 			return 0;
   4721 		}
   4722 
   4723 		//WeaponTypeClass const * weapon = Techno_Type_Class()->PrimaryWeapon;
   4724 		//BulletTypeClass const * bullet = weapon->Bullet;
   4725 		//WarheadTypeClass const * warhead = weapon->WarheadPtr;
   4726 		WeaponTypeClass const * weapon = &Weapons[weapon_type];
   4727 		const BulletTypeClass &bullet = BulletTypeClass::As_Reference(Weapons[weapon_type].Fires);
   4728 		WarheadTypeClass const * warhead = &Warheads[bullet.Warhead];
   4729 
   4730 		int mrange = min(weapon->Range, 0x0400);
   4731 
   4732 		int value = ((weapon->Attack * warhead->Modifier[ARMOR_STEEL]) * mrange * warhead->SpreadFactor) / weapon->ROF;
   4733 		//if (Techno_Type_Class()->Is_Two_Shooter()) {
   4734 		//	value *= 2;
   4735 		//}
   4736 		if (bullet.IsInaccurate) {
   4737 			value /= 2;
   4738 		}
   4739 		return(value/50);
   4740 	}
   4741 	return(0);
   4742 }
   4743 
   4744 
   4745 /***********************************************************************************************
   4746  * TechnoClass::Anti_Infantry -- Calculates the anti-infantry strength of this object.         *
   4747  *                                                                                             *
   4748  *    This routine is used to determine the anti-infantry strength of this object. The         *
   4749  *    typical user of this routine is the expert system base defense AI.                       *
   4750  *                                                                                             *
   4751  * INPUT:   none                                                                               *
   4752  *                                                                                             *
   4753  * OUTPUT:  Returns with the anti-infantry strength of this object. The value returned is      *
   4754  *          abstract and should only be used for relative comparisons.                         *
   4755  *                                                                                             *
   4756  * WARNINGS:   none                                                                            *
   4757  *                                                                                             *
   4758  * HISTORY:                                                                                    *
   4759  *   10/02/1995 JLB : Created.                                                                 *
   4760  *=============================================================================================*/
   4761 int TechnoClass::Anti_Infantry(void) const
   4762 {
   4763 	//assert(IsActive);
   4764 
   4765 	if (Is_Weapon_Equipped()) {
   4766 		//if (!Techno_Type_Class()->PrimaryWeapon->Bullet->IsAntiGround) return(0);
   4767 		WeaponType weapon_type = Techno_Type_Class()->Primary;
   4768 		if (weapon_type != WEAPON_NONE && BulletTypeClass::As_Reference(Weapons[weapon_type].Fires).IsAntiAircraft) {
   4769 			return 0;
   4770 		}
   4771 
   4772 		//WeaponTypeClass const * weapon = Techno_Type_Class()->PrimaryWeapon;
   4773 		//BulletTypeClass const * bullet = weapon->Bullet;
   4774 		//WarheadTypeClass const * warhead = weapon->WarheadPtr;
   4775 		WeaponTypeClass const * weapon = &Weapons[weapon_type];
   4776 		const BulletTypeClass &bullet = BulletTypeClass::As_Reference(Weapons[weapon_type].Fires);
   4777 		WarheadTypeClass const * warhead = &Warheads[bullet.Warhead];
   4778 		int mrange = min(weapon->Range, 0x0400);
   4779 
   4780 		int value = ((weapon->Attack * warhead->Modifier[ARMOR_NONE]) * mrange * warhead->SpreadFactor) / weapon->ROF;
   4781 		//if (Techno_Type_Class()->Is_Two_Shooter()) {
   4782 		//	value *= 2;
   4783 		//}
   4784 		if (bullet.IsInaccurate) {
   4785 			value /= 2;
   4786 		}
   4787 		return(value/50);
   4788 	}
   4789 	return(0);
   4790 }
   4791 
   4792 
   4793 
   4794 int TechnoTypeClass::Legal_Placement(CELL pos) const
   4795 {
   4796 	if (pos == -1) return(0);
   4797 
   4798 	/*
   4799 	**	Normal buildings must check to see that every foundation square is free of
   4800 	**	obstacles. If this check passes for all foundation squares, only then does the
   4801 	**	routine return that it is legal to place.
   4802 	*/
   4803 	short const * offset = Occupy_List(true);
   4804 	bool build = (What_Am_I() == RTTI_BUILDINGTYPE);
   4805 
   4806 	if (build) {
   4807 		return ((BuildingTypeClass*)this)->Legal_Placement(pos);
   4808 	}
   4809 
   4810 	while (offset != NULL && *offset != REFRESH_EOL) {
   4811 		CELL cell = pos + *offset++;
   4812 		if (!Map.In_Radar(cell)) return(false);
   4813 		//if (build) {
   4814 		//	if (!Map[cell].Is_Clear_To_Build(Speed)) {
   4815 		//		return(0);
   4816 		//	}
   4817 		//} else {
   4818 			//if (!Map[cell].Is_Clear_To_Move(Speed, false, false)) {
   4819 			if (!Map[cell].Is_Clear_To_Move(false, false)) {
   4820 				return(0);
   4821 			}
   4822 		//}
   4823 	}
   4824 	return(1);
   4825 }
   4826 #endif //USE_RA_AI