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