BULLET.CPP (37640B)
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\bullet.cpv 2.18 16 Oct 1995 16:50:00 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 : BULLET.CPP * 24 * * 25 * Programmer : Joe L. Bostic * 26 * * 27 * Start Date : April 23, 1994 * 28 * * 29 * Last Update : March 18, 1995 [JLB] * 30 * * 31 *---------------------------------------------------------------------------------------------* 32 * Functions: * 33 * BulletClass::AI -- Logic processing for bullet. * 34 * BulletClass::As_Target -- Converts the bullet into a target value. * 35 * BulletClass::BulletClass -- Bullet constructor. * 36 * BulletClass::BulletClass -- Default constructor for bullet objects. * 37 * BulletClass::Detach -- Removes specified target from this bullet's targeting system. * 38 * BulletClass::Draw_It -- Displays the bullet at location specified. * 39 * BulletClass::Init -- Clears the bullets array for scenario preparation. * 40 * BulletClass::Mark -- Performs related map refreshing under bullet. * 41 * BulletClass::Occupy_List -- Determines the bullet occupation list. * 42 * BulletClass::Unlimbo -- Transitions a bullet object into the game render/logic system. * 43 * BulletClass::delete -- Bullet memory delete. * 44 * BulletClass::new -- Allocates memory for bullet object. * 45 * BulletClass::Validate -- validates bullet pointer * 46 * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ 47 48 #include "function.h" 49 50 #define GRAVITY 3 51 52 /* 53 ** This contains the value of the Virtual Function Table Pointer 54 */ 55 void * BulletClass::VTable; 56 57 58 /*********************************************************************************************** 59 * BulletClass::Validate -- validates bullet pointer * 60 * * 61 * INPUT: * 62 * none. * 63 * * 64 * OUTPUT: * 65 * 1 = ok, 0 = error * 66 * * 67 * WARNINGS: * 68 * none. * 69 * * 70 * HISTORY: * 71 * 08/09/1995 BRR : Created. * 72 *=============================================================================================*/ 73 #ifdef CHEAT_KEYS 74 int BulletClass::Validate(void) const 75 { 76 int num; 77 78 num = Bullets.ID(this); 79 if (num < 0 || num >= BULLET_MAX) { 80 Validate_Error("BULLET"); 81 return (0); 82 } 83 else 84 return (1); 85 } 86 #else 87 #define Validate() 88 #endif 89 90 91 /*********************************************************************************************** 92 * BulletClass::BulletClass -- Default constructor for bullet objects. * 93 * * 94 * This is the default constructor for bullet objects. A bullet constructed by this routine * 95 * is not in a usable state for game purposes. It must be constructed by the full * 96 * (parameterized) constructor -- usually called as part of the overloaded "new" operator. * 97 * * 98 * INPUT: none * 99 * * 100 * OUTPUT: none * 101 * * 102 * WARNINGS: Do not use bullets that were constructed solely by this routine. * 103 * * 104 * HISTORY: * 105 * 09/24/1994 JLB : Created. * 106 *=============================================================================================*/ 107 BulletClass::BulletClass(void) : 108 Class(0) 109 { 110 Payback = NULL; 111 IsToAnimate = false; 112 Altitude = 0; 113 Riser = 0; 114 TarCom = TARGET_NONE; 115 Strength = 0; 116 IsLocked = true; 117 IsInaccurate = false; 118 } 119 120 121 /*********************************************************************************************** 122 * BulletClass::new -- Allocates memory for bullet object. * 123 * * 124 * This function will "allocate" a block of memory for a bullet object. * 125 * This memory block is merely lifted from a fixed pool of blocks. * 126 * * 127 * INPUT: size -- The size of the memory block needed. * 128 * * 129 * OUTPUT: Returns with a pointer to an available bullet object block. * 130 * * 131 * WARNINGS: none * 132 * * 133 * HISTORY: * 134 * 05/02/1994 JLB : Created. * 135 *=============================================================================================*/ 136 void * BulletClass::operator new(size_t ) 137 { 138 void * ptr = Bullets.Allocate(); 139 if (ptr) { 140 ((BulletClass *)ptr)->Set_Active(); 141 } 142 return(ptr); 143 } 144 145 146 /*********************************************************************************************** 147 * BulletClass::delete -- Bullet memory delete. * 148 * * 149 * Since bullets memory is merely "allocated" out of a pool, it never * 150 * actually gets deleted. * 151 * * 152 * INPUT: ptr -- Generic pointer to bullet object. * 153 * * 154 * OUTPUT: none * 155 * * 156 * WARNINGS: none * 157 * * 158 * HISTORY: * 159 * 05/02/1994 JLB : Created. * 160 *=============================================================================================*/ 161 void BulletClass::operator delete(void *ptr) 162 { 163 if (ptr) { 164 ((BulletClass *)ptr)->IsActive = false; 165 } 166 Bullets.Free((BulletClass *)ptr); 167 } 168 169 170 /*********************************************************************************************** 171 * BulletClass::BulletClass -- Bullet constructor. * 172 * * 173 * This is the constructor for the bullet class. It handles all * 174 * initialization of the bullet and starting it in motion toward its * 175 * target. * 176 * * 177 * INPUT: id -- The type of bullet this is (could be missile). * 178 * * 179 * OUTPUT: none * 180 * * 181 * WARNINGS: none * 182 * * 183 * HISTORY: * 184 * 05/02/1994 JLB : Created. * 185 * 06/20/1994 JLB : Firer is a base class pointer. * 186 * 12/10/1994 JLB : Auto calculate range optional. * 187 * 12/12/1994 JLB : Handles small arms as an instantaneous effect. * 188 * 12/23/1994 JLB : Fixed scatter algorithm for non-homing projectiles. * 189 * 12/31/1994 JLB : Removed range parameter (not needed). * 190 *=============================================================================================*/ 191 BulletClass::BulletClass(BulletType id) : 192 Class(&BulletTypeClass::As_Reference(id)) 193 { 194 Altitude = 0; 195 IsInaccurate = false; 196 IsLocked = true; 197 // IsLocked = false; 198 IsToAnimate = false; 199 Payback = 0; 200 Riser = 0; 201 Strength = Class->MaxStrength; 202 TarCom = TARGET_NONE; 203 } 204 205 206 /*********************************************************************************************** 207 * BulletClass::Occupy_List -- Determines the bullet occupation list. * 208 * * 209 * This function will determine the cell occupation list and return a pointer to it. Most * 210 * bullets are small and the list is usually short, but on occasion, it can be a list that * 211 * rivals the size of regular vehicles. * 212 * * 213 * INPUT: none * 214 * * 215 * OUTPUT: Returns with a pointer to the cell offset list that covers all the cells a bullet * 216 * is over. * 217 * * 218 * WARNINGS: none * 219 * * 220 * HISTORY: * 221 * 06/20/1994 JLB : Created. * 222 * 01/05/1995 JLB : Handles projectiles with altitude. * 223 *=============================================================================================*/ 224 short const *BulletClass::Occupy_List(void) const 225 { 226 Validate(); 227 switch (*this) { 228 case BULLET_FLAME: 229 return(Coord_Spillage_List(Coord, 25)); 230 231 case BULLET_NUKE_UP: 232 case BULLET_NUKE_DOWN: 233 return(Coord_Spillage_List(Coord, 48)); 234 235 default: 236 if (Altitude) { 237 static CELL _list[10]; 238 const short * ptr = Coord_Spillage_List(Coord, 5); 239 int index = 0; 240 CELL cell1 = Coord_Cell(Coord); 241 242 while (ptr[index] != REFRESH_EOL) { 243 _list[index] = ptr[index]; 244 index++; 245 } 246 247 COORDINATE coord = XY_Coord(0, Altitude); 248 coord = Coord_Sub(Coord, coord); 249 CELL cell2 = Coord_Cell(coord); 250 ptr = Coord_Spillage_List(coord, 5); 251 while (*ptr != REFRESH_EOL) { 252 _list[index++] = (*ptr++) + (cell2 - cell1); 253 } 254 _list[index] = REFRESH_EOL; 255 return(_list); 256 } 257 } 258 return(Coord_Spillage_List(Coord, 10)); 259 } 260 261 262 /*********************************************************************************************** 263 * BulletClass::Mark -- Performs related map refreshing under bullet. * 264 * * 265 * This routine marks the objects under the bullet so that they will * 266 * be redrawn. This is necessary as the bullet moves -- objects under * 267 * its path need to be restored. * 268 * * 269 * INPUT: none * 270 * * 271 * OUTPUT: none * 272 * * 273 * WARNINGS: none * 274 * * 275 * HISTORY: * 276 * 05/02/1994 JLB : Created. * 277 *=============================================================================================*/ 278 bool BulletClass::Mark(MarkType mark) 279 { 280 Validate(); 281 if (ObjectClass::Mark(mark)) { 282 if (!Class->IsInvisible) { 283 Map.Refresh_Cells(Coord_Cell(Coord), Occupy_List()); 284 } 285 return(true); 286 } 287 return(false); 288 } 289 290 291 /*********************************************************************************************** 292 * BulletClass::AI -- Logic processing for bullet. * 293 * * 294 * This routine will perform all logic (flight) logic on the bullet. * 295 * Primarily this is motion, fuse tracking, and detonation logic. Call * 296 * this routine no more than once per bullet per game tick. * 297 * * 298 * INPUT: none * 299 * * 300 * OUTPUT: none * 301 * * 302 * WARNINGS: none * 303 * * 304 * HISTORY: * 305 * 05/02/1994 JLB : Created. * 306 *=============================================================================================*/ 307 void BulletClass::AI(void) 308 { 309 Validate(); 310 COORDINATE coord; 311 312 ObjectClass::AI(); 313 314 /* 315 ** Balistic objects are handled here. 316 */ 317 bool forced = false; // Forced explosion. 318 if (Class->IsArcing) { 319 Altitude += Riser; 320 321 if (Altitude <= 0) { 322 forced = true; 323 } 324 if (Riser > -100) { 325 Riser -= GRAVITY; 326 } 327 } 328 if (Class->IsDropping) { 329 Altitude += Riser; 330 331 if (Altitude <= 0) { 332 forced = true; 333 } 334 if (Riser > -100) { 335 Riser -= 1; 336 } 337 } 338 339 /* 340 ** Homing projectiles constantly change facing to face toward the target but 341 ** they only do so every other game frame (improves game speed and makes 342 ** missiles not so deadly). 343 */ 344 if ((Frame & 0x01) && Class->IsHoming && Target_Legal(TarCom)) { 345 PrimaryFacing.Set_Desired(Direction256(Coord, ::As_Coord(TarCom))); 346 } 347 348 /* 349 ** Move the projectile forward according to its speed 350 ** and direction. 351 */ 352 coord = Coord; 353 if (Class->IsFlameEquipped) { 354 if (IsToAnimate) { 355 new AnimClass(ANIM_SMOKE_PUFF, coord, 1); 356 } 357 IsToAnimate = !IsToAnimate; 358 } 359 360 /* 361 ** Handle any body rotation at this time. This process must 362 ** occur every game fame in order to achieve smooth rotation. 363 */ 364 if (PrimaryFacing.Is_Rotating()) { 365 PrimaryFacing.Rotation_Adjust(Class->ROT); 366 } 367 368 switch (Physics(coord, PrimaryFacing)) { 369 370 /* 371 ** When a projectile reaches the edge of the world, it 372 ** vanishes from existence -- presumed to explode off 373 ** map. 374 */ 375 case IMPACT_EDGE: 376 // if (IsLocked) { 377 Mark(); 378 Delete_This(); 379 // } 380 break; 381 382 default: 383 case IMPACT_NONE: 384 385 /* 386 ** The projectile has moved. Check its fuse. If detonation 387 ** is signaled, then do so. Otherwise, just move. 388 */ 389 case IMPACT_NORMAL: 390 Mark(); 391 // IsLocked = true; 392 if (!Class->IsHigh) { 393 CellClass * cellptr = &Map[Coord_Cell(coord)]; 394 if (cellptr->Overlay != OVERLAY_NONE && OverlayTypeClass::As_Reference(cellptr->Overlay).IsHigh) { 395 forced = true; 396 Coord = coord = Cell_Coord(Coord_Cell(coord)); 397 } 398 } 399 400 /* 401 ** Bullets are generaly more effective when they are fired at aircraft. 402 */ 403 if (Class->IsAntiAircraft && As_Aircraft(TarCom) && Distance(TarCom) < 0x0080) { 404 forced = true; 405 406 if (*this == BULLET_TOW) { 407 Strength += Strength/3; 408 } else { 409 Strength += Strength/2; 410 } 411 } 412 413 if (!forced && (Class->IsDropping || !Fuse_Checkup(coord))) { 414 Coord = coord; 415 Mark(); 416 417 /* 418 ** Certain projectiles loose strength when they travel. 419 */ 420 if (*this == BULLET_BULLET) { 421 if (Strength > 5) Strength--; 422 } 423 424 } else { 425 426 /* 427 ** When the target is reached, explode and do the damage 428 ** required of it. For homing objects, don't force the explosion to 429 ** match the target position. Non-homing projectiles adjust position so 430 ** that they hit the target. This compensates for the error in line of 431 ** flight logic. 432 */ 433 Mark(); 434 if (!forced && !Class->IsArcing && !Class->IsHoming && Fuse_Target()) { 435 Coord = Fuse_Target(); 436 } 437 438 /* 439 ** Non-aircraft targets apply damage to the ground. 440 */ 441 if (!Is_Target_Aircraft(TarCom) || As_Aircraft(TarCom)->In_Which_Layer() == LAYER_GROUND) { 442 Explosion_Damage(Coord, Strength, Payback, Class->Warhead); 443 } else { 444 445 /* 446 ** Special damage apply for SAM missiles. This is the only way that missile 447 ** damage affects the aircraft target. 448 */ 449 if (Distance(TarCom) < 0x0080) { 450 AircraftClass * object = As_Aircraft(TarCom); 451 452 int str = Strength; 453 if (object) object->Take_Damage(str, 0, Class->Warhead, Payback); 454 } 455 } 456 457 /* 458 ** For projectiles that are invisible while travelling toward the target, 459 ** allow scatter effect for the impact animation. 460 */ 461 if (Class->IsInvisible) { 462 Coord = Coord_Scatter(Coord, 0x0020); 463 } 464 if (Class->Explosion != ANIM_NONE) { 465 AnimClass * newanim = new AnimClass(Class->Explosion, Coord); 466 if (newanim) { 467 newanim->Sort_Above(TarCom); 468 } 469 470 // MBL 05.20.2020 471 // Fix for Nuke or Atom Bomb killing structures and units during animation sequence and not getting kills tracked 472 // Per https://jaas.ea.com/browse/TDRA-6610 473 // 474 if (newanim && Class->Explosion == ANIM_ATOM_BLAST && newanim->OwnerHouse == HOUSE_NONE) { 475 if (Payback && Payback->House && Payback->House->Class) { 476 newanim->Set_Owner(Payback->House->Class->House); 477 } 478 } 479 } 480 Delete_This(); 481 return; 482 } 483 break; 484 } 485 } 486 487 488 /*********************************************************************************************** 489 * BulletClass::Draw_It -- Displays the bullet at location specified. * 490 * * 491 * This routine displays the bullet visual at the location specified. * 492 * * 493 * INPUT: x,y -- The center coordinate to render the bullet at. * 494 * * 495 * window -- The window to clip to. * 496 * * 497 * OUTPUT: none * 498 * * 499 * WARNINGS: none * 500 * * 501 * HISTORY: * 502 * 06/20/1994 JLB : Created. * 503 * 06/27/1994 JLB : Takes a window clipping parameter. * 504 * 01/08/1995 JLB : Handles translucent colors if necessary. * 505 *=============================================================================================*/ 506 void BulletClass::Draw_It(int x, int y, WindowNumberType window) 507 { 508 Validate(); 509 int facing = Facing_To_32(PrimaryFacing); 510 511 /* 512 ** Certain projectiles aren't visible. This includes small bullets (which are actually 513 ** invisible) and flame thrower flames (which are rendered as an animation instead of a projectile). 514 */ 515 if (Class->IsInvisible) return; 516 517 /* 518 ** If there is no shape loaded for this object, then 519 ** it obviously can't be rendered -- just bail. 520 */ 521 void const * shapeptr = Class->Get_Image_Data(); 522 if (!shapeptr) return; 523 524 /* 525 ** Get the basic shape number for this projectile. 526 */ 527 int shapenum = 0; 528 if (!Class->IsFaceless) { 529 shapenum = UnitClass::BodyShape[facing]; 530 } 531 532 /* 533 ** For tumbling projectiles, fetch offset stage. 534 */ 535 if (*this == BULLET_NAPALM) { 536 shapenum += Frame % 6; 537 } 538 539 if (*this == BULLET_GRENADE) { 540 shapenum += Frame % 8; 541 // Timer++; 542 } 543 544 /* 545 ** For flying projectiles, draw the shadow and adjust the actual projectile body 546 ** render position. 547 */ 548 if (Altitude) { 549 CC_Draw_Shape(shapeptr, shapenum, x, y, window, SHAPE_PREDATOR|SHAPE_CENTER|SHAPE_WIN_REL|SHAPE_FADING, NULL, Map.FadingShade); 550 y -= Lepton_To_Pixel(Altitude); 551 } 552 553 /* 554 ** Draw the main body of the projectile. 555 */ 556 ShapeFlags_Type flags = SHAPE_NORMAL; 557 if (Class->IsTranslucent) { 558 flags = SHAPE_GHOST; 559 } 560 CC_Draw_Shape(this, shapeptr, shapenum, x, y, window, flags|SHAPE_CENTER|SHAPE_WIN_REL, NULL, Map.UnitShadow); 561 } 562 563 564 /*********************************************************************************************** 565 * BulletClass::Init -- Clears the bullets array for scenario preparation. * 566 * * 567 * This routine will zero out the bullet tracking list and object array in preparation for * 568 * the start of a new scenario. All bullets cease to exists after this function is * 569 * called. * 570 * * 571 * INPUT: none * 572 * * 573 * OUTPUT: none * 574 * * 575 * WARNINGS: none * 576 * * 577 * HISTORY: * 578 * 08/15/1994 JLB : Created. * 579 *=============================================================================================*/ 580 void BulletClass::Init(void) 581 { 582 BulletClass *ptr; 583 584 Bullets.Free_All(); 585 586 ptr = new BulletClass(); 587 VTable = ((void **)(((char *)ptr) + sizeof(AbstractClass) - 4))[0]; 588 delete ptr; 589 } 590 591 592 /*********************************************************************************************** 593 * BulletClass::Detach -- Removes specified target from this bullet's targeting system. * 594 * * 595 * When an object is removed from the game system, it must be removed all targeting and * 596 * tracking systems as well. This routine is used to remove the specified object from the * 597 * bullet. If the object isn't part of this bullet's tracking system, then no action is * 598 * performed. * 599 * * 600 * INPUT: target -- The target to remove from this tracking system. * 601 * * 602 * all -- Is the target going away for good as opposed to just cloaking/hiding? * 603 * * 604 * OUTPUT: none * 605 * * 606 * WARNINGS: none * 607 * * 608 * HISTORY: * 609 * 09/24/1994 JLB : Created. * 610 *=============================================================================================*/ 611 void BulletClass::Detach(TARGET target, bool all) 612 { 613 Validate(); 614 ObjectClass * obj = As_Object(target); 615 616 if (obj == Payback) { 617 Payback = 0; 618 } 619 620 if (all && target == TarCom) { 621 TarCom = TARGET_NONE; 622 } 623 } 624 625 626 /*********************************************************************************************** 627 * BulletClass::Unlimbo -- Transitions a bullet object into the game render/logic system. * 628 * * 629 * This routine is used to take a bullet object that is in limbo and transition it to the * 630 * game system. A bullet object so transitioned, will be drawn and logic processing * 631 * performed. In effect, it comes into existance. * 632 * * 633 * INPUT: coord -- The location where the bullet object is to appear. * 634 * * 635 * dir -- The initial facing for the bullet object. * 636 * * 637 * OUTPUT: bool; Was the unlimbo successful? * 638 * * 639 * WARNINGS: none * 640 * * 641 * HISTORY: * 642 * 01/10/1995 JLB : Created. * 643 *=============================================================================================*/ 644 bool BulletClass::Unlimbo(COORDINATE coord, DirType dir) 645 { 646 Validate(); 647 /* 648 ** Try to unlimbo the bullet as far as the base class is concerned. Use the already 649 ** set direction and strength if the "punt" values were passed in. This allows a bullet 650 ** to be setup prior to being launched. 651 */ 652 if (ObjectClass::Unlimbo(coord)) { 653 COORDINATE tcoord = As_Coord(TarCom); 654 655 /* 656 ** Homing projectiles (missiles) do NOT override facing. They just fire in the 657 ** direction specified and let the chips fall where they may. 658 */ 659 if (!Class->IsHoming && !Class->IsDropping) { 660 dir = Direction(tcoord); 661 } 662 663 /* 664 ** Possibly adjust the target if this projectile is inaccurate. This occurs whenever 665 ** certain weapons are trained upon targets they were never designed to attack. Example: when 666 ** turrets or anti-tank missiles are fired at infantry. Indirect 667 ** fire is inherently inaccurate. 668 */ 669 if (IsInaccurate || Class->IsInaccurate || 670 ((Is_Target_Cell(TarCom) || Is_Target_Infantry(TarCom)) && (Class->Warhead == WARHEAD_AP || Class->IsFueled))) { 671 672 /* 673 ** Inaccuracy for low velocity or homing projectiles manifests itself as a standard 674 ** Cicular Error of Probability (CEP) algorithm. High speed projectiles usually 675 ** just overshoot the target by extending the straight line flight. 676 */ 677 if (Class->IsHoming || Class->IsArcing) { 678 int scatterdist = ::Distance(coord, tcoord)/3; 679 scatterdist = MIN(scatterdist, 0x0200); 680 681 if (*this == BULLET_GRENADE) { 682 scatterdist = ::Distance(coord, tcoord)/4; 683 scatterdist = MIN(scatterdist, 0x0080); 684 } 685 686 dir = (DirType)((dir + (Random_Pick(0, 10)-5)) & 0x00FF); 687 tcoord = Coord_Scatter(tcoord, Random_Pick(0, scatterdist)); 688 } else { 689 tcoord = Coord_Move(tcoord, dir, Random_Pick(0, 0x0100)); 690 } 691 692 /* 693 ** Limit scatter to the weapon range of the firer. 694 */ 695 if (Payback) { 696 if (!Payback->In_Range(tcoord, 0) && !Payback->In_Range(tcoord, 1)) { 697 tcoord = Coord_Move(tcoord, ::Direction(tcoord, Coord), Distance(tcoord) - MAX(Payback->Weapon_Range(0), Payback->Weapon_Range(1))); 698 } 699 } 700 } 701 702 /* 703 ** For very fast and invisible projectiles, just make the projectile exist at the target 704 ** location and dispense with the actual flight. 705 */ 706 if (Class->MaxSpeed == MPH_LIGHT_SPEED && Class->IsInvisible) { 707 Coord = tcoord; 708 } 709 710 /* 711 ** Set the range equal to either the class defined range or the calculated 712 ** number of game frames it would take for the projectile to reach the target. 713 */ 714 int range = 0xFF; 715 if (!Class->Range) { 716 if (!Class->IsDropping) { 717 range = (::Distance(tcoord, Coord) / Class->MaxSpeed) + 4; 718 } 719 } else { 720 range = Class->Range; 721 } 722 723 /* 724 ** Projectile speed is usually the default value for that projectile, but 725 ** certian projectiles alter speed according to the distance to the 726 ** target. 727 */ 728 int speed = Class->MaxSpeed; 729 if (speed == MPH_LIGHT_SPEED) speed = MPH_IMMOBILE; 730 if (Class->IsArcing) { 731 speed = Class->MaxSpeed + (Distance(tcoord)>>5); 732 733 /* 734 ** Set minimum speed (i.e., distance) for arcing projectiles. 735 */ 736 speed = MAX(speed, 25); 737 } 738 if (!Class->IsDropping) { 739 Fly_Speed(255, (MPHType)speed); 740 } 741 742 /* 743 ** Arm the fuse. 744 */ 745 Arm_Fuse(Coord, tcoord, range, ((As_Aircraft(TarCom)!=0) ? 0 : Class->Arming)); 746 747 /* 748 ** Projectiles that make a ballistic flight to impact point must determine a 749 ** vertical component for the projectile launch. This is crudely simulated 750 ** by biasing ground speed according to target distance and then giving 751 ** enough vertical velocity to keep the projectile airborne for the 752 ** desired amount of time. The mathematically correct solution would be to 753 ** calculate launch angle (given fixed projectile velocity) and then derive 754 ** the vertical and horizontal components. This solution would require a 755 ** of square root and an arcsine lookup table. OUCH! 756 */ 757 Altitude = 0; 758 Riser = 0; 759 if (Class->IsArcing) { 760 Altitude = 1; 761 Riser = ((Distance(tcoord)/2) / (speed+1))*GRAVITY; 762 Riser = MAX(Riser, (signed char)10); 763 } 764 if (Class->IsDropping) { 765 Altitude = Pixel_To_Lepton(24); 766 Riser = 0; 767 } 768 769 PrimaryFacing = dir; 770 return(true); 771 } 772 return(false); 773 } 774 775 776 /*********************************************************************************************** 777 * BulletClass::As_Target -- Converts the bullet into a target value. * 778 * * 779 * This support routine is used to convert the bullet (as a pointer) into a target * 780 * value (which is a number). * 781 * * 782 * INPUT: none * 783 * * 784 * OUTPUT: Returns the bullet as a target value. * 785 * * 786 * WARNINGS: none * 787 * * 788 * HISTORY: * 789 * 09/08/1994 JLB : Created. * 790 *=============================================================================================*/ 791 TARGET BulletClass::As_Target(void) const 792 { 793 Validate(); 794 return(Build_Target(KIND_BULLET, Bullets.ID(this))); 795 }