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