home *** CD-ROM | disk | FTP | other *** search
/ Quaaake Level & Editor 2 / Quaaake_2.iso / Quake / cujo12 / SOURCE.ZIP / CUJOAI.QC < prev    next >
Encoding:
Text File  |  1996-10-24  |  63.3 KB  |  2,036 lines

  1. /*╔═══════════════════════════════════════════════════════════════════════╗
  2.   ║                                                                       ║
  3.   ║                            CUJO BOT ver 1.2                           ║
  4.   ║                                                                       ║
  5.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  6.  
  7. // distance from player before cujo initiates a teleport
  8. float   teleport_dist  = 1500;
  9.  
  10. // Prototypes
  11.  
  12. float   ()                                             CUJO_FindTarget; //
  13. void    ()                                             CUJO_ai_stand;   //
  14. void    ()                                             CUJO_pain;       //
  15. void    ()                                             CUJO_die;        //
  16. void    ()                                             CUJO_SightSound; //
  17. void    ()                                             CUJO_FoundTarget;//
  18. void    ()                                             CUJO_HuntTarget; //
  19. void    (float dist)                                   CUJO_ai_walk;    //
  20. void    ()                                             CUJO_ai_turn;    //
  21. void    (float dist)                                   CUJO_ai_run;     //
  22. void    (float dist)                                   CUJO_ai_follow;  //
  23. void    ()                                             CUJO_ai_face;
  24. void    (float d)                                      CUJO_ai_charge;
  25. void    (void () thinkst)                              CUJO_CheckRefire;//
  26. void    (float deathframe)                             CUJO_SelfDeactivate;//
  27. void    ()                                             CUJO_EatGib;
  28. void    ()                                             CUJO_EatHead;
  29. entity  ()                                             CUJO_FindFood;
  30.  
  31. void    ()                                             CUJO_TeleportToOwner;//
  32. vector  ()                                             CUJO_SpawnPos;
  33. float   (vector SpawnPos)                              CUJO_CheckSpawnPos;
  34.  
  35. float   goal_range;
  36. float   goal_vis;
  37. float   goal_infront;
  38. float   goal_yaw;
  39.  
  40. // Cujo Prototypes - called by player
  41.  
  42. void () CUJO_Precache;
  43. void () CUJO_Activate;        // Activate Cujo
  44. void () CUJO_Deactivate;    // DeActivate Cujo
  45. void () CUJO_Toggle;        // Toggle Cujo on and off
  46. void () CUJO_AttackToggle;    // Toggle Cujo's auto-firing in auto mode
  47. /*
  48. void ()    CUJO_Attack;        // Have Cujo fire at its current target in auto mode
  49. */
  50. void ()    CUJO_TeleportHome;    // Have Cujo teleport back to its owner
  51. void ()    CUJO_LightToggle;    // Have Cujo teleport back to its owner
  52. void () CUJO_SetDogView;
  53. void () CUJO_SetPlayerView;
  54.  
  55.  
  56.  
  57. $cd /raid/quake/id1/models/dog
  58. $origin 0 0 24
  59. $base base
  60. $skin skin
  61.  
  62. $frame attack1 attack2 attack3 attack4 attack5 attack6 attack7 attack8
  63.  
  64. $frame death1 death2 death3 death4 death5 death6 death7 death8 death9
  65.  
  66. $frame deathb1 deathb2 deathb3 deathb4 deathb5 deathb6 deathb7 deathb8
  67. $frame deathb9
  68.  
  69. $frame pain1 pain2 pain3 pain4 pain5 pain6
  70.  
  71. $frame painb1 painb2 painb3 painb4 painb5 painb6 painb7 painb8 painb9 painb10
  72. $frame painb11 painb12 painb13 painb14 painb15 painb16
  73.  
  74. $frame run1 run2 run3 run4 run5 run6 run7 run8 run9 run10 run11 run12
  75.  
  76. $frame leap1 leap2 leap3 leap4 leap5 leap6 leap7 leap8 leap9
  77.  
  78. $frame stand1 stand2 stand3 stand4 stand5 stand6 stand7 stand8 stand9
  79.  
  80. $frame walk1 walk2 walk3 walk4 walk5 walk6 walk7 walk8
  81.  
  82.  
  83. void() dog_leap1;
  84. void() dog_run1;
  85.  
  86.  
  87. /*╔═══════════════════════════════════════════════════════════════════════╗
  88.   ║                                                                       ║
  89.   ║                 Routines called by Cujo, self = Cujo                  ║
  90.   ║                                                                       ║
  91.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  92.  
  93. /*╔═══════════════════════════════════════════════════════════════════════╗
  94.   ║                                                                       ║
  95.   ║ CUJO_PrintCujoStatus                                                  ║
  96.   ║                                                                       ║
  97.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  98.  
  99. void () CUJO_PrintCujoStatus =
  100.  
  101. {
  102.   if (!deathmatch)
  103.     sprint (self, "Cujo is not here.\n");
  104.   else
  105.   {
  106.     if (self.Cujo_avail)
  107.       sprint (self, "Cujo is not here.\n");
  108.     else
  109.       sprint (self, "Cujo is not available.\n");
  110.   }
  111. };
  112.  
  113. /*╔═══════════════════════════════════════════════════════════════════════╗
  114.   ║                                                                       ║
  115.   ║ CUJO_ResetGoalEntity                                                  ║
  116.   ║                                                                       ║
  117.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  118.  
  119. void () CUJO_ResetGoalEntity =
  120.  
  121. {
  122.   self.oldenemy = self.enemy;
  123.   self.enemy = world;
  124.  
  125.   self.goalentity = self.owner;
  126.   self.movetarget = self.owner;
  127. };
  128.  
  129. /*╔═══════════════════════════════════╗
  130.   ║                                   ║
  131.   ║ Cujo_bite                         ║
  132.   ║                                   ║
  133.   ╚═══════════════════════════════════╝*/
  134.  
  135. void() CUJO_bite =
  136. {
  137.   local vector    delta;
  138.   local float     ldmg;
  139.  
  140.   if (!self.enemy) return;
  141.  
  142.   CUJO_ai_charge(10);
  143.  
  144.   if (!CanDamage (self.enemy, self)) return;
  145.  
  146.   delta = self.enemy.origin - self.origin;
  147.  
  148.   if (vlen(delta) > 100) return;
  149.  
  150.   // does twice the damage of a normal dog and has a random chance
  151.   // of gibbing!
  152.  
  153.   if ((self.enemy.health < 10) && (random () < 0.1))
  154.   {
  155.     {
  156.       ldmg = 50 * random ();
  157.     }
  158.   }
  159.   else
  160.   {
  161.     ldmg = (random() + random() + random()) * 16;
  162.   }
  163.  
  164.   T_Damage (self.enemy, self, self, ldmg);
  165. };
  166.  
  167. /*╔═══════════════════════════════════╗
  168.   ║                                   ║
  169.   ║ Cujo_jumptouch                    ║
  170.   ║                                   ║
  171.   ╚═══════════════════════════════════╝*/
  172.  
  173. void()    CUJO_JumpTouch =
  174. {
  175.   local    float    ldmg;
  176.  
  177.   if (self.health <= 0) return;
  178.  
  179.   if (other.takedamage)
  180.   {
  181.     if ( vlen(self.velocity) > 300 )
  182.     {
  183.       // Cujo now has a chance to gib on a jumping attack
  184.       ldmg = 20 + 20*random();
  185.  
  186.       T_Damage (other, self, self, ldmg);
  187.     }
  188.   }
  189.  
  190.   if (!checkbottom(self))
  191.   {
  192.     if (self.flags & FL_ONGROUND)
  193.     {    // jump randomly to not get hung up
  194.       //dprint ("popjump\n");
  195.       self.touch = SUB_Null;
  196.       self.think = dog_leap1;
  197.       self.nextthink = time + 0.1;
  198.     }
  199.  
  200.     return;    // not on ground yet
  201.   }
  202.  
  203.   self.touch = SUB_Null;
  204.   self.think = dog_run1;
  205.   self.nextthink = time + 0.1;
  206. };
  207.  
  208.  
  209. void() CUJO_stand1    =[    $stand1,    CUJO_stand2    ] {CUJO_ai_stand();};
  210. void() CUJO_stand2    =[    $stand2,    CUJO_stand3    ] {CUJO_ai_stand();};
  211. void() CUJO_stand3    =[    $stand3,    CUJO_stand4    ] {CUJO_ai_stand();};
  212. void() CUJO_stand4    =[    $stand4,    CUJO_stand5    ] {CUJO_ai_stand();};
  213. void() CUJO_stand5    =[    $stand5,    CUJO_stand6    ] {CUJO_ai_stand();};
  214. void() CUJO_stand6    =[    $stand6,    CUJO_stand7    ] {CUJO_ai_stand();};
  215. void() CUJO_stand7    =[    $stand7,    CUJO_stand8    ] {CUJO_ai_stand();};
  216. void() CUJO_stand8    =[    $stand8,    CUJO_stand9    ] {CUJO_ai_stand();};
  217. void() CUJO_stand9    =[    $stand9,    CUJO_stand1    ] {CUJO_ai_stand();};
  218.  
  219. /*╔═══════════════════════════════════╗
  220.   ║                                   ║
  221.   ║ Cujo_walk1                        ║
  222.   ║                                   ║
  223.   ╚═══════════════════════════════════╝*/
  224.  
  225.  
  226. void() CUJO_walk1    =[    $walk1 ,    CUJO_walk2    ]
  227. {
  228.   if (random() < 0.2)
  229.     sound (self, CHAN_VOICE, "dog/idle.wav", 1, ATTN_IDLE);
  230.  
  231.   CUJO_ai_walk (8);
  232. };
  233.  
  234.  
  235. void() CUJO_walk2    =[    $walk2 ,    CUJO_walk3    ] {CUJO_ai_walk(8);};
  236. void() CUJO_walk3    =[    $walk3 ,    CUJO_walk4    ] {CUJO_ai_walk(8);};
  237. void() CUJO_walk4    =[    $walk4 ,    CUJO_walk5    ] {CUJO_ai_walk(8);};
  238. void() CUJO_walk5    =[    $walk5 ,    CUJO_walk6    ] {CUJO_ai_walk(8);};
  239. void() CUJO_walk6    =[    $walk6 ,    CUJO_walk7    ] {CUJO_ai_walk(8);};
  240. void() CUJO_walk7    =[    $walk7 ,    CUJO_walk8    ] {CUJO_ai_walk(8);};
  241. void() CUJO_walk8    =[    $walk8 ,    CUJO_walk1    ] {CUJO_ai_walk(8);};
  242.  
  243. /*╔═══════════════════════════════════╗
  244.   ║                                   ║
  245.   ║ Cujo_run1                         ║
  246.   ║                                   ║
  247.   ╚═══════════════════════════════════╝*/
  248.  
  249. void() CUJO_run1        =[    $run1  ,    CUJO_run2    ]
  250. {
  251.   if (random() < 0.2)
  252.     sound (self, CHAN_VOICE, "dog/idle.wav", 1, ATTN_IDLE);
  253.  
  254.   CUJO_ai_run (16);
  255. };
  256.  
  257. void() CUJO_run2        =[    $run2  ,    CUJO_run3    ] {CUJO_ai_run(32);};
  258. void() CUJO_run3        =[    $run3  ,    CUJO_run4    ] {CUJO_ai_run(32);};
  259. void() CUJO_run4        =[    $run4  ,    CUJO_run5    ] {CUJO_ai_run(20);};
  260. void() CUJO_run5        =[    $run5  ,    CUJO_run6    ] {CUJO_ai_run(64);};
  261. void() CUJO_run6        =[    $run6  ,    CUJO_run7    ] {CUJO_ai_run(32);};
  262. void() CUJO_run7        =[    $run7  ,    CUJO_run8    ] {CUJO_ai_run(16);};
  263. void() CUJO_run8        =[    $run8  ,    CUJO_run9    ] {CUJO_ai_run(32);};
  264. void() CUJO_run9        =[    $run9  ,    CUJO_run10    ] {CUJO_ai_run(32);};
  265. void() CUJO_run10    =[    $run10  ,    CUJO_run11    ] {CUJO_ai_run(20);};
  266. void() CUJO_run11    =[    $run11  ,    CUJO_run12    ] {CUJO_ai_run(64);};
  267. void() CUJO_run12    =[    $run12  ,    CUJO_run1    ] {CUJO_ai_run(32);};
  268.  
  269. void()    CUJO_follow1    =[    $run1  ,    CUJO_follow2    ]
  270. {
  271.   if (random() < 0.2)
  272.     sound (self, CHAN_VOICE, "dog/idle.wav", 1, ATTN_IDLE);
  273.  
  274.   CUJO_ResetGoalEntity ();
  275.  
  276.   CUJO_ai_follow (18);
  277. };
  278. void()    CUJO_follow2    =[    $run2  ,    CUJO_follow3    ] {CUJO_ai_follow (32);};
  279. void()    CUJO_follow3    =[    $run3  ,    CUJO_follow4    ] {CUJO_ai_follow (32);};
  280. void()    CUJO_follow4    =[    $run4  ,    CUJO_follow5    ] {CUJO_ai_follow (20);};
  281. void()    CUJO_follow5    =[    $run5  ,    CUJO_follow6    ] {CUJO_ai_follow (64);};
  282. void()    CUJO_follow6    =[    $run6  ,    CUJO_follow7    ] {CUJO_ai_follow (32);};
  283. void()    CUJO_follow7    =[    $run7  ,    CUJO_follow8    ] {CUJO_ai_follow (16);};
  284. void()    CUJO_follow8    =[    $run8  ,    CUJO_follow9    ] {CUJO_ai_follow (32);};
  285. void()  CUJO_follow9    =[    $run9  ,    CUJO_follow10    ] {CUJO_ai_follow(32);};
  286. void()  CUJO_follow10    =[    $run10  ,    CUJO_follow11    ] {CUJO_ai_follow(20);};
  287. void()  CUJO_follow11    =[    $run11  ,    CUJO_follow12    ] {CUJO_ai_follow(64);};
  288. void()  CUJO_follow12    =[    $run12  ,    CUJO_follow1    ] {CUJO_ai_follow(32);};
  289.  
  290.  
  291. /*╔═══════════════════════════════════╗
  292.   ║                                   ║
  293.   ║ Cujo_atta1                        ║
  294.   ║                                   ║
  295.   ╚═══════════════════════════════════╝*/
  296.  
  297.  
  298. void() CUJO_atta1    =[    $attack1,    CUJO_atta2    ] {ai_charge(10);};
  299. void() CUJO_atta2    =[    $attack2,    CUJO_atta3    ] {ai_charge(10);};
  300. void() CUJO_atta3    =[    $attack3,    CUJO_atta4    ] {ai_charge(10);};
  301.  
  302. void() CUJO_atta4    =[    $attack4,    CUJO_atta5    ]
  303. {
  304.   sound (self, CHAN_VOICE, "dog/dattack1.wav", 1, ATTN_NORM);
  305.   CUJO_bite();
  306. };
  307.  
  308. void() CUJO_atta5    =[    $attack5,    CUJO_atta6    ] {ai_charge(10);};
  309. void() CUJO_atta6    =[    $attack6,    CUJO_atta7    ] {ai_charge(10);};
  310. void() CUJO_atta7    =[    $attack7,    CUJO_atta8    ] {ai_charge(10);};
  311. void() CUJO_atta8    =[    $attack8,    CUJO_run1       ]
  312. {
  313.   local   float   r;
  314.  
  315.   if (self.goalentity.classname == "dog_food")
  316.   {
  317.     remove (self.goalentity);
  318.     self.health = self.health + 15;
  319.  
  320.     if ((deathmatch) && (self.health > 100))
  321.       self.health = 100;
  322.     else if (self.health > 200)
  323.       self.health = 200;
  324.  
  325.     r = random ();
  326.  
  327.     if (r < 0.25)
  328.       sprint (self.owner, "Cujo likes his Gibby-chow!\n");
  329.     else if (r < 0.5)
  330.       sprint (self.owner, "Dogs love a Gibby-Treat!\n");
  331.     else if (r < 0.75)
  332.       sprint (self.owner, "Cujo ate some gibs.\n");
  333.     else
  334.       sprint (self.owner, "Cujo's eating his enemies again...\n");
  335.  
  336.   }
  337.   else
  338.     CUJO_ai_charge (10);
  339.  
  340.   CUJO_CheckRefire (CUJO_atta1);
  341. };
  342.  
  343. /*╔═══════════════════════════════════╗
  344.   ║                                   ║
  345.   ║ Cujo_leap1                        ║
  346.   ║                                   ║
  347.   ╚═══════════════════════════════════╝*/
  348.  
  349. void() CUJO_leap1    =[    $leap1,        CUJO_leap2    ] {ai_face();};
  350. void() CUJO_leap2    =[    $leap2,        CUJO_leap3    ]
  351. {
  352.   CUJO_ai_face();
  353.   self.touch = CUJO_JumpTouch;
  354.  
  355.   makevectors (self.angles);
  356.  
  357.   self.origin_z = self.origin_z + 1;
  358.   self.velocity = v_forward * 300 + '0 0 200';
  359.  
  360.   if (self.flags & FL_ONGROUND)
  361.     self.flags = self.flags - FL_ONGROUND;
  362. };
  363.  
  364. void() CUJO_leap3    =[    $leap3,        CUJO_leap4    ] {};
  365. void() CUJO_leap4    =[    $leap4,        CUJO_leap5    ] {};
  366. void() CUJO_leap5    =[    $leap5,        CUJO_leap6    ] {};
  367. void() CUJO_leap6    =[    $leap6,        CUJO_leap7    ] {};
  368. void() CUJO_leap7    =[    $leap7,        CUJO_leap8    ] {};
  369. void() CUJO_leap8    =[    $leap8,        CUJO_leap9    ] {};
  370. void() CUJO_leap9    =[    $leap9,        CUJO_leap9    ] {};
  371.  
  372. /*╔═══════════════════════════════════╗
  373.   ║                                   ║
  374.   ║ Cujo_pain1                        ║
  375.   ║                                   ║
  376.   ╚═══════════════════════════════════╝*/
  377.  
  378. void() CUJO_pain1    =[    $pain1 ,    CUJO_pain2    ] {};
  379. void() CUJO_pain2    =[    $pain2 ,    CUJO_pain3    ] {};
  380. void() CUJO_pain3    =[    $pain3 ,    CUJO_pain4    ] {};
  381. void() CUJO_pain4    =[    $pain4 ,    CUJO_pain5    ] {};
  382. void() CUJO_pain5    =[    $pain5 ,    CUJO_pain6    ] {};
  383. void() CUJO_pain6    =[    $pain6 ,    CUJO_run1    ] {};
  384.  
  385. void() CUJO_painb1    =[    $painb1 ,    CUJO_painb2    ] {};
  386. void() CUJO_painb2    =[    $painb2 ,    CUJO_painb3    ] {};
  387. void() CUJO_painb3    =[    $painb3 ,    CUJO_painb4    ] {ai_pain(4);};
  388. void() CUJO_painb4    =[    $painb4 ,    CUJO_painb5    ] {ai_pain(12);};
  389. void() CUJO_painb5    =[    $painb5 ,    CUJO_painb6    ] {ai_pain(12);};
  390. void() CUJO_painb6    =[    $painb6 ,    CUJO_painb7    ] {ai_pain(2);};
  391. void() CUJO_painb7    =[    $painb7 ,    CUJO_painb8    ] {};
  392. void() CUJO_painb8    =[    $painb8 ,    CUJO_painb9    ] {ai_pain(4);};
  393. void() CUJO_painb9    =[    $painb9 ,    CUJO_painb10    ] {};
  394. void() CUJO_painb10    =[    $painb10 ,    CUJO_painb11    ] {ai_pain(10);};
  395. void() CUJO_painb11    =[    $painb11 ,    CUJO_painb12    ] {};
  396. void() CUJO_painb12    =[    $painb12 ,    CUJO_painb13    ] {};
  397. void() CUJO_painb13    =[    $painb13 ,    CUJO_painb14    ] {};
  398. void() CUJO_painb14    =[    $painb14 ,    CUJO_painb15    ] {};
  399. void() CUJO_painb15    =[    $painb15 ,    CUJO_painb16    ] {};
  400. void() CUJO_painb16    =[    $painb16 ,    CUJO_run1    ] {};
  401.  
  402. void() CUJO_pain =
  403. {
  404.   sound (self, CHAN_VOICE, "dog/dpain1.wav", 1, ATTN_NORM);
  405.  
  406.   if (random() > 0.5)
  407.     CUJO_pain1 ();
  408.   else
  409.     CUJO_painb1 ();
  410. };
  411.  
  412. /*╔═══════════════════════════════════╗
  413.   ║                                   ║
  414.   ║ Cujo_die                          ║
  415.   ║                                   ║
  416.   ║ Cujo deactivates self in last     ║
  417.   ║ frame                             ║
  418.   ║                                   ║
  419.   ╚═══════════════════════════════════╝*/
  420.  
  421. void() CUJO_die1        =[    $death1,    CUJO_die2    ] {};
  422. void() CUJO_die2        =[    $death2,    CUJO_die3    ] {};
  423. void() CUJO_die3        =[    $death3,    CUJO_die4    ] {};
  424. void() CUJO_die4        =[    $death4,    CUJO_die5    ] {};
  425. void() CUJO_die5        =[    $death5,    CUJO_die6    ] {};
  426. void() CUJO_die6        =[    $death6,    CUJO_die7    ] {};
  427. void() CUJO_die7        =[    $death7,    CUJO_die8    ] {};
  428. void() CUJO_die8        =[    $death8,    CUJO_die9    ] {};
  429. void() CUJO_die9        =[    $death9,    CUJO_die9    ] {CUJO_SelfDeactivate (0);};
  430.  
  431. // used to set the dead-entity body to the correct final death frame
  432. // then set up the next think to make the body disappear
  433. void() CUJO_die10        =[    $death9,    CUJO_die10    ]
  434. {
  435.   self.nextthink = time + 120;  // wait 2 minutes till body disappears
  436.   self.think = SUB_Remove;
  437. };
  438.  
  439. void() CUJO_dieb1        =[    $deathb1,    CUJO_dieb2    ] {};
  440. void() CUJO_dieb2        =[    $deathb2,    CUJO_dieb3    ] {};
  441. void() CUJO_dieb3        =[    $deathb3,    CUJO_dieb4    ] {};
  442. void() CUJO_dieb4        =[    $deathb4,    CUJO_dieb5    ] {};
  443. void() CUJO_dieb5        =[    $deathb5,    CUJO_dieb6    ] {};
  444. void() CUJO_dieb6        =[    $deathb6,    CUJO_dieb7    ] {};
  445. void() CUJO_dieb7        =[    $deathb7,    CUJO_dieb8    ] {};
  446. void() CUJO_dieb8        =[    $deathb8,    CUJO_dieb9    ] {};
  447. void() CUJO_dieb9        =[    $deathb9,    CUJO_dieb9    ] {CUJO_SelfDeactivate (1);};
  448.  
  449. // used to set the dead-entity body to the correct final death frame
  450. // then set up the next think to make the body disappear
  451. void() CUJO_dieb10            =[    $deathb9,    CUJO_dieb10    ]
  452. {
  453.   self.nextthink = time + 120;  // wait 2 minutes till body disappears
  454.   self.think = SUB_Remove;
  455. };
  456.  
  457.  
  458. void() CUJO_die =
  459. {
  460. // check for gib
  461.   if (self.health < -35)
  462.   {
  463.     sound (self, CHAN_VOICE, "player/udeath.wav", 1, ATTN_NORM);
  464.     ThrowGib ("progs/gib3.mdl", self.health);
  465.     ThrowGib ("progs/gib3.mdl", self.health);
  466.     ThrowGib ("progs/gib3.mdl", self.health);
  467.  
  468.     ThrowHead ("progs/h_cujo.mdl", self.health);
  469.  
  470.     // Cujo heads only lay around for 2 minutes, then POOF!
  471.     self.nextthink = time + 120;
  472.     self.think = SUB_Remove;
  473.  
  474.     CUJO_SelfDeactivate ();
  475.  
  476.     return;
  477.   }
  478.  
  479. // regular death
  480.   sound (self, CHAN_VOICE, "dog/ddeath.wav", 1, ATTN_NORM);
  481.   self.solid = SOLID_NOT;
  482.  
  483.   if (random() > 0.5)
  484.     CUJO_die1 ();
  485.   else
  486.     CUJO_dieb1 ();
  487. };
  488.  
  489. /*╔═══════════════════════════════════════════════════════════════════════╗
  490.   ║                                                                       ║
  491.   ║ Cujo_checkmelee                                                       ║
  492.   ║                                                                       ║
  493.   ║ Returns TRUE if a melee attack would hit right now                    ║
  494.   ║                                                                       ║
  495.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  496.  
  497. float()    CUJO_CheckMelee =
  498. {
  499.   local   float   dist;
  500.  
  501.   dist = vlen (self.goalentity.origin - self.origin);
  502.  
  503.   if ( ((self.goalentity.classname == "dog_food") && (dist < 100)) ||
  504.       ((self.goalentity.classname != "dog_food") && (dist < 50)) )
  505. //  if (goal_range == RANGE_MELEE)
  506.   {
  507.     self.attack_state = AS_MELEE;
  508.  
  509. //    bprint ("CUJO_CheckMelee says TRUE!\n");
  510.  
  511.     return TRUE;
  512.   }
  513.  
  514.   return FALSE;
  515. };
  516.  
  517. /*╔═══════════════════════════════════════════════════════════════════════╗
  518.   ║                                                                       ║
  519.   ║ Cujo_checkjump                                                        ║
  520.   ║                                                                       ║
  521.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  522.  
  523. float()    CUJO_CheckJump =
  524. {
  525.   local    vector    dist;
  526.   local    float    d;
  527.  
  528.   if (self.origin_z + self.mins_z > self.goalentity.origin_z + self.goalentity.mins_z
  529.   + 0.75 * self.goalentity.size_z)
  530.     return FALSE;
  531.  
  532.   if (self.origin_z + self.maxs_z < self.goalentity.origin_z + self.goalentity.mins_z
  533.   + 0.25 * self.goalentity.size_z)
  534.     return FALSE;
  535.  
  536.   dist = self.goalentity.origin - self.origin;
  537.   dist_z = 0;
  538.  
  539.   d = vlen(dist);
  540.  
  541.   if (d < 80) return FALSE;
  542.  
  543.   if (d > 150) return FALSE;
  544.  
  545.   return TRUE;
  546. };
  547.  
  548. /*╔═══════════════════════════════════════════════════════════════════════╗
  549.   ║                                                                       ║
  550.   ║ Cujo_checkattack                                                      ║
  551.   ║                                                                       ║
  552.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  553.  
  554. // this checks the attack against self.goalentity not self.enemy!
  555.  
  556. float()    CUJO_CheckAttack =
  557. {
  558.   local    vector    vec;
  559.  
  560.   if ((((self.goalentity == self.owner) || (self.goalentity.classname == "monster_zombie"))
  561.       || (self.Cujo_attack == FALSE)) && (self.goalentity.classname != "dog_food"))
  562.   {
  563.     CUJO_ResetGoalEntity ();
  564.  
  565.     self.think = self.th_stand;
  566.     self.nextthink = time + 0.1;
  567.  
  568.     return FALSE;
  569.   }
  570.  
  571.   if (CUJO_CheckMelee ())
  572.   {
  573.     self.attack_state = AS_MELEE;
  574.     return TRUE;
  575.   }
  576.  
  577.   if (self.goalentity.classname != "dog_food")
  578.   {
  579.     if (CUJO_CheckJump ())
  580.     {
  581.       self.attack_state = AS_MISSILE;
  582.       return TRUE;
  583.     }
  584.   }
  585.  
  586.   return FALSE;
  587.  
  588. };
  589.  
  590. /*╔═══════════════════════════════════════════════════════════════════════╗
  591.   ║                                                                       ║
  592.   ║ CUJO_ai_run_melee                                                     ║
  593.   ║                                                                       ║
  594.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  595.  
  596. void() CUJO_ai_run_melee =
  597. {
  598.   CUJO_ai_face ();
  599.  
  600.   if (FacingIdeal())
  601.   {
  602.     self.th_melee ();
  603.     self.attack_state = AS_STRAIGHT;
  604.   }
  605. };
  606.  
  607. /*╔═══════════════════════════════════════════════════════════════════════╗
  608.   ║                                                                       ║
  609.   ║ CUJO_ai_run_missile                                                   ║
  610.   ║                                                                       ║
  611.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  612.  
  613. void() CUJO_ai_run_missile =
  614. {
  615.   CUJO_ai_face ();
  616.  
  617.   if (FacingIdeal())
  618.   {
  619.     self.th_missile ();
  620.     self.attack_state = AS_STRAIGHT;
  621.   }
  622. };
  623.  
  624. /*╔═══════════════════════════════════════════════════════════════════════╗
  625.   ║                                                                       ║
  626.   ║ Cujo_ai_run                                                           ║
  627.   ║                                                                       ║
  628.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  629.  
  630. void(float dist) CUJO_ai_run =
  631. {
  632.   local    vector    delta;
  633.   local    float    axis;
  634.   local    float    direct, ang_rint, ang_floor, ang_ceil, f_temp;
  635.   local string  str_temp;
  636.  
  637. //  sprint (self.owner, "-- enter ai_run --\n");
  638.  
  639.   movedist = dist;
  640.  
  641. //  sprint (self.owner, "Enemy is ");
  642. //  sprint (self.owner, self.enemy.classname);
  643. //  sprint (self.owner, "\n");
  644. //  sprint (self.owner, "Goalentity is ");
  645. //  sprint (self.owner, self.goalentity.classname);
  646. //  sprint (self.owner, "\n");
  647.  
  648.   if ((self.goalentity.classname != "dog_food") && (self.goalentity.health <= 0))
  649.   {
  650.     CUJO_ResetGoalEntity ();
  651.  
  652.     //sprint (self.owner, "CUJO_ai_run: reset goalentity\n");
  653.  
  654.     if (self.oldenemy.health > 0)
  655.     {
  656.       self.enemy = self.oldenemy;
  657.       self.goalentity = self.oldenemy;
  658.  
  659.       CUJO_HuntTarget ();
  660.     }
  661.     else
  662.     {
  663.       if (CUJO_FindTarget()) return;
  664.       else
  665.       {
  666.         CUJO_ResetGoalEntity ();
  667.  
  668.         //sprint (self.owner, "CUJO_ai_run #2: reset goalentity\n");
  669.  
  670.         self.th_walk ();
  671.  
  672. //        sprint (self.owner, "-- Left ai_run at return 1 --\n");
  673.         return;
  674.       }
  675.     }
  676.  
  677. //    sprint (self.owner, "-- Left ai_run at return 2 --\n");
  678.     return;
  679.   }
  680.  
  681.   if (self.goalentity.classname != "dog_food")
  682.     self.show_hostile = time + 1;        // wake up other monsters
  683.  
  684.   goal_vis = visible(self.goalentity);
  685.   if (goal_vis) self.search_time = time + 5;
  686.  
  687.   goal_infront = infront (self.goalentity);
  688.   goal_range = range (self.goalentity);
  689.   goal_yaw = vectoyaw (self.goalentity.origin - self.origin);
  690.  
  691.   if ((self.attack_state == AS_MISSILE) && (self.goalentity.classname != "dog_food"))
  692.   {
  693.     CUJO_ai_run_missile ();
  694.     return;
  695.   }
  696.   if (self.attack_state == AS_MELEE)
  697.   {
  698.  
  699. // for debugging
  700. //    f_temp = vlen (self.origin - self.goalentity.origin);
  701. //    str_temp = ftos (f_temp);
  702.  
  703. //    sprint (self.owner, "Cujo is ");
  704. //    sprint (self.owner, str_temp);
  705. //    sprint (self.owner, " units from ");
  706. //    sprint (self.owner, self.goalentity.classname);
  707. //    sprint (self.owner, ".\n");
  708.  
  709.     CUJO_ai_run_melee ();
  710.  
  711.     return;
  712.   }
  713.  
  714.   if (CUJO_CheckAttack ()) return;    // beginning an attack
  715.  
  716.   movetogoal (dist);        // done in C code...
  717.  
  718. //  sprint (self.owner, "-- exit ai_run --\n");
  719. };
  720.  
  721. /*╔═══════════════════════════════════════════════════════════════════════╗
  722.   ║                                                                       ║
  723.   ║ Cujo_SightSound                                                       ║
  724.   ║                                                                       ║
  725.   ║ dog stands in place until target acquired or paustime is exceeded     ║
  726.   ║                                                                       ║
  727.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  728.  
  729. void() CUJO_SightSound =
  730. {
  731.   sound (self, CHAN_VOICE, "dog/dsight.wav", 1, ATTN_NORM);
  732. };
  733.  
  734. /*╔═══════════════════════════════════════════════════════════════════════╗
  735.   ║                                                                       ║
  736.   ║ Cujo_HuntTarget                                                       ║
  737.   ║                                                                       ║
  738.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  739.  
  740. void() CUJO_HuntTarget =
  741. {
  742. // for debugging
  743.   //sprint (self.owner, "CUJO_HuntTarget: ");
  744.   //sprint (self.owner, self.goalentity.classname);
  745.   //sprint (self.owner, "\n");
  746.  
  747.   self.think = self.th_run;
  748.   self.ideal_yaw = vectoyaw(self.goalentity.origin - self.origin);
  749.   self.nextthink = time + 0.1;
  750.  
  751.   SUB_AttackFinished (0.1);
  752. };
  753.  
  754. /*╔═══════════════════════════════════════════════════════════════════════╗
  755.   ║                                                                       ║
  756.   ║ Cujo_FoundTarget                                                      ║
  757.   ║                                                                       ║
  758.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  759.  
  760. void() CUJO_FoundTarget =
  761. {
  762.   local float   f_dist;
  763.   local string  s_dist;
  764.   local float   r;
  765.  
  766.   f_dist = vlen (self.goalentity.origin - self.origin);
  767.   s_dist = ftos (f_dist);
  768.  
  769.   self.show_hostile = time + 1;        // wake up other monsters
  770.  
  771.   r = random ();
  772.  
  773.   if (r < 0.5)
  774.     sprint (self.owner, "rrrrRROOF!\n");
  775.   else
  776.     sprint (self.owner, "grrrrllll...\n");
  777.  
  778. // for debugging
  779.   //sprint (self.owner, "Cujo seeking target");
  780.   //sprint (self.owner, self.goalentity.classname);
  781.   //sprint (self.owner, "\n");
  782.  
  783.   CUJO_SightSound ();
  784.  
  785.   CUJO_HuntTarget ();
  786. };
  787.  
  788. /*╔═══════════════════════════════════════════════════════════════════════╗
  789.   ║                                                                       ║
  790.   ║ CUJO_FindFood                                                         ║
  791.   ║                                                                       ║
  792.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  793.  
  794. // returns the first visible gib or head within Cujo's range
  795.  
  796. entity () CUJO_FindFood =
  797. {
  798.   local entity   head, selected;
  799.   local entity   food;
  800.   local float    dist;
  801.  
  802.   // check for any gibs or heads visible to Cujo
  803.   dist = 1000;
  804.   selected = world;
  805.  
  806.   head = findradius (self.origin, dist);
  807.  
  808.   while (head)
  809.   {
  810.     if ((head.classname == "dog_food") && (visible (head)) &&
  811.         (trace_plane_dist < dist) && (head != self) && (head != self.owner))
  812.     {
  813.       selected = head;
  814.       dist = trace_plane_dist;
  815.     }
  816.  
  817.     head = head.chain;
  818.   }
  819.  
  820.   return selected;
  821. };
  822.  
  823. /*╔═══════════════════════════════════════════════════════════════════════╗
  824.   ║                                                                       ║
  825.   ║ Cujo_FindTarget                                                       ║
  826.   ║                                                                       ║
  827.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  828.  
  829. float() CUJO_FindTarget =
  830. {
  831.   local entity   head, selected;
  832.   local entity   food;
  833.   local float    dist;
  834.  
  835.   // check for any visible gibs or heads within Cujo's search radius
  836.   food = CUJO_FindFood ();
  837.  
  838.   // check for any enemies visible to Cujo
  839.   dist = 1000;
  840.   selected = world;
  841.  
  842.   head = findradius(self.origin, dist);
  843.  
  844.   while(head)
  845.   {
  846.     if(!(head.flags & FL_NOTARGET) && ((head.flags & FL_CLIENT) || (head.flags & FL_MONSTER)))
  847.       if (((teamplay) &&  (head.team != self.owner.team)) || (!teamplay))
  848.         if ((head.health > 0) && (head != self) && (head != self.owner))
  849.           if ((visible(head)) && (head.classname != "monster_zombie"))
  850.           if (trace_plane_dist < dist)
  851.           {
  852.             selected = head;
  853.             dist = trace_plane_dist;
  854.           }
  855.  
  856.     head = head.chain;
  857.   }
  858.  
  859.   self.enemy = selected;
  860.  
  861.   if ((food != world) && (food != self.owner) && (self.Cujo_attack)
  862.     && ((self.health < 50) || ((deathmatch) && (self.health < 25))))
  863.   {
  864.     self.goalentity = food;
  865.     self.enemy = food;
  866.  
  867.     CUJO_FoundTarget ();
  868.  
  869.     return TRUE;
  870.   }
  871.   else if ((self.enemy == world) || (self.enemy == self.owner))
  872.   {
  873.     if (((food != world) && (food != self.owner)) && (self.health < 191) &&
  874.         (self.Cujo_attack))
  875.     {
  876.       self.goalentity = food;
  877.       self.enemy = food;
  878.  
  879.       CUJO_FoundTarget ();
  880.  
  881.       return TRUE;
  882.     }
  883.  
  884.     return FALSE;
  885.   }
  886.   else  // an enemy target has been sighted
  887.   {
  888.     if (!(self.Cujo_attack))  // if not supposed to attack, just growl
  889.     {
  890.       if (random () < 0.10)
  891.       {
  892.         // make Cujo bark or growl
  893.         if (random () < 0.5)
  894.           sound (self, CHAN_VOICE, "dog/dsight.wav", 1, ATTN_NORM);
  895.         else
  896.           sound (self, CHAN_VOICE, "dog/dattack1.wav", 1, ATTN_NORM);
  897.       }
  898.  
  899.       CUJO_ResetGoalEntity ();
  900.  
  901.       return FALSE;
  902.     }
  903.  
  904.     // if Cujo is staying, make him attack!
  905.     self.Cujo_stay = FALSE;
  906.  
  907.     self.goalentity = self.enemy;
  908.  
  909.     CUJO_FoundTarget();
  910.  
  911.     return TRUE;
  912.   }
  913. };
  914.  
  915. /*╔═══════════════════════════════════════════════════════════════════════╗
  916.   ║                                                                       ║
  917.   ║ CUJO_ai_face                                                          ║
  918.   ║                                                                       ║
  919.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  920.  
  921. void() CUJO_ai_face =
  922. {
  923.   self.ideal_yaw = vectoyaw (self.goalentity.origin - self.origin);
  924.   ChangeYaw ();
  925. };
  926.  
  927. /*╔═══════════════════════════════════════════════════════════════════════╗
  928.   ║                                                                       ║
  929.   ║ CUJO_ai_charge                                                        ║
  930.   ║                                                                       ║
  931.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  932.  
  933. void (float d) CUJO_ai_charge =
  934. {
  935.   CUJO_ai_face ();
  936.  
  937.   movetogoal (d);        // done in C code...
  938. };
  939.  
  940. /*╔═══════════════════════════════════════════════════════════════════════╗
  941.   ║                                                                       ║
  942.   ║ Cujo_ai_stand                                                         ║
  943.   ║                                                                       ║
  944.   ║ dog stands in place until target acquired or paustime is exceeded     ║
  945.   ║                                                                       ║
  946.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  947.  
  948. void() CUJO_ai_stand =
  949. {
  950.   if (CUJO_FindTarget ()) return;
  951.  
  952.   if (vlen (self.origin - self.owner.origin) > teleport_dist)
  953.   {
  954.     CUJO_TeleportToOwner ();
  955.   }
  956.   else if (vlen (self.origin - self.owner.origin) > 100)
  957.   {
  958.     // if Cujo is staying, then he shouldn't run in place!
  959.     if (self.Cujo_stay == TRUE) return;
  960.  
  961.     CUJO_follow1 ();
  962.  
  963.     return;
  964.   }
  965. };
  966.  
  967. /*╔═══════════════════════════════════════════════════════════════════════╗
  968.   ║                                                                       ║
  969.   ║ Cujo_ai_walk                                                          ║
  970.   ║                                                                       ║
  971.   ║ Cujo is walking and searching for targets                             ║
  972.   ║                                                                       ║
  973.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  974.  
  975. void(float dist) CUJO_ai_walk =
  976. {
  977.   if (CUJO_FindTarget ()) return;
  978.  
  979.   if (vlen (self.origin - self.owner.origin) > 100)
  980.   {
  981.     CUJO_follow1();
  982.  
  983.     return;
  984.   }
  985.   else if (vlen (self.origin - self.owner.origin) > teleport_dist)
  986.   {
  987.     CUJO_TeleportToOwner ();
  988.   }
  989.  
  990.   movetogoal (dist);  // this is done in C code
  991. };
  992.  
  993. /*╔═══════════════════════════════════════════════════════════════════════╗
  994.   ║                                                                       ║
  995.   ║ Cujo_ai_follow                                                        ║
  996.   ║                                                                       ║
  997.   ║ Cujo is following player and searching for targets                    ║
  998.   ║                                                                       ║
  999.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1000.  
  1001. void(float dist) CUJO_ai_follow =
  1002. {
  1003.   if (CUJO_FindTarget ()) return;
  1004.  
  1005.   //sprint (self.owner, "Following owner.\n");
  1006.   //sprint (self.owner, "Goalentity is ");
  1007.   //sprint (self.owner, self.goalentity.classname);
  1008.   //sprint (self.owner, "\n");
  1009.   //sprint (self.owner, "Enemy is ");
  1010.   //sprint (self.owner, self.enemy.classname);
  1011.   //sprint (self.owner, "\n");
  1012.   //sprint (self.owner, "Movetarget is ");
  1013.   //sprint (self.owner, self.movetarget.classname);
  1014.   //sprint (self.owner, "\n");
  1015.  
  1016.   // if Cujo is in STAY mode then do not follow the player!
  1017.  
  1018.   if ((vlen (self.origin - self.owner.origin) <= 100) || (self.Cujo_stay))
  1019.   {
  1020.     self.pausetime = time + 2;
  1021.  
  1022.     self.th_stand ();
  1023.  
  1024.     return;
  1025.   }
  1026.   else if (vlen (self.origin - self.owner.origin) > teleport_dist)
  1027.   {
  1028.     CUJO_TeleportToOwner ();
  1029.   }
  1030.  
  1031.   movetogoal (dist);  // done in C code
  1032. };
  1033.  
  1034. /*╔═══════════════════════════════════════════════════════════════════════╗
  1035.   ║                                                                       ║
  1036.   ║ Cujo_ai_turn                                                          ║
  1037.   ║                                                                       ║
  1038.   ║ Turn Cujo towards ideal_yaw if no enemies were sighted                ║
  1039.   ║                                                                       ║
  1040.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1041.  
  1042. void() CUJO_ai_turn =
  1043. {
  1044.   if (CUJO_FindTarget ()) return;
  1045.  
  1046.   ChangeYaw ();
  1047. };
  1048.  
  1049. /*╔═══════════════════════════════════════════════════════════════════════╗
  1050.   ║                                                                       ║
  1051.   ║ Cujo_CheckRefire                                                      ║
  1052.   ║                                                                       ║
  1053.   ║ Determine if the enemy is still visible for attack                    ║
  1054.   ║                                                                       ║
  1055.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1056.  
  1057. void (void () thinkst) CUJO_CheckRefire =
  1058. {
  1059.   if (!visible (self.goalentity) || (self.goalentity.health <= 0))
  1060.   {
  1061.     CUJO_ResetGoalEntity ();
  1062.  
  1063.     self.think = self.th_stand;
  1064.     self.nextthink = time + 0.1;
  1065.  
  1066.     return;
  1067.   }
  1068.  
  1069.   self.think = thinkst;  // was thinkst
  1070. };
  1071.  
  1072. /*╔═══════════════════════════════════════════════════════════════════════╗
  1073.   ║                                                                       ║
  1074.   ║ CUJO_CorpsePain                                                       ║
  1075.   ║                                                                       ║
  1076.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1077.  
  1078. void () CUJO_CorpsePain =
  1079. {
  1080.  SpawnMeatSpray (self.origin, crandom() * 100 * v_right);
  1081. };
  1082.  
  1083. /*╔═══════════════════════════════════════════════════════════════════════╗
  1084.   ║                                                                       ║
  1085.   ║ CUJO_CorpseDie;                                                       ║
  1086.   ║                                                                       ║
  1087.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1088.  
  1089. void () CUJO_CorpseDie =
  1090. {
  1091.   local   float   i;
  1092.   local   float   j;
  1093.  
  1094.   sound (self, CHAN_VOICE, "player/udeath.wav", 1, ATTN_NORM);
  1095.  
  1096.   ThrowHead ("progs/h_cujo.mdl", self.health);
  1097.  
  1098.   i = 0;
  1099.  
  1100.   while(i<3)
  1101.   {
  1102.     j = random();
  1103.  
  1104.     if(j > 0.6)
  1105.       ThrowGib ("progs/gib1.mdl", self.health);
  1106.     else if(j > 0.3)
  1107.       ThrowGib ("progs/gib2.mdl", self.health);
  1108.     else
  1109.       ThrowGib ("progs/gib3.mdl", self.health);
  1110.  
  1111.     i = i + 1;
  1112.   }
  1113.  
  1114.   // Cujo's head only stays around for 2 minutes...
  1115.   self.nextthink = time + 120;
  1116.   self.think = SUB_Remove;
  1117. };
  1118.  
  1119. /*╔═══════════════════════════════════════════════════════════════════════╗
  1120.   ║                                                                       ║
  1121.   ║ Cujo_SelfDeactivate                                                   ║
  1122.   ║                                                                       ║
  1123.   ║ Cujo deactivates himself (when he is killed, basically)               ║
  1124.   ║                                                                       ║
  1125.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1126.  
  1127. // deathframe = 1 = dog death 1
  1128. //            = 2 = dog death 2
  1129. //            = 3 = gibbed, don't remove entity, because it's a head now
  1130.  
  1131. void (float deathframe) CUJO_SelfDeactivate =
  1132. {
  1133.   local   entity  bot_owner;
  1134.   local   entity  Cujobody;
  1135.   local   entity  temp_self;
  1136.  
  1137.   bot_owner = self.owner;
  1138.  
  1139.   // create a dummy Cujo body... this alleviates the problem of the
  1140.   // original Cujo entity becoming lost if the player reactivates Cujo
  1141.   // before the body has disappeared
  1142.  
  1143.   if (deathframe != 3)
  1144.   {
  1145.     Cujobody = spawn();
  1146.  
  1147.     Cujobody.angles = self.angles;
  1148.     Cujobody.classname = "cujo_body";
  1149.     Cujobody.owner=self.owner;
  1150.     Cujobody.view_ofs = '0 0 15';
  1151.     Cujobody.team = self.team;
  1152.  
  1153.     Cujobody.health = 30;
  1154.     Cujobody.solid = SOLID_BBOX;
  1155.     Cujobody.takedamage = DAMAGE_AIM;
  1156.     Cujobody.movetype = MOVETYPE_STEP;
  1157.  
  1158.     Cujobody.th_stand = SUB_Null;
  1159.     Cujobody.th_walk = SUB_Null;
  1160.     Cujobody.th_run = SUB_Null;
  1161.     Cujobody.th_pain = CUJO_CorpsePain;
  1162.     Cujobody.th_melee = SUB_Null;
  1163.     Cujobody.th_missile = SUB_Null;
  1164.     Cujobody.th_die = CUJO_CorpseDie;
  1165.     Cujobody.touch = SUB_Null;
  1166.  
  1167.     setmodel (Cujobody, "progs/cujo.mdl");
  1168.     setsize (Cujobody, '-16 -16 -24', '16 16 0');
  1169.     setorigin(Cujobody, self.origin);
  1170.  
  1171.     // on next think, set the new body to the correct death frame
  1172.     // this is the only way I figured out to do it...  sometimes you'll
  1173.     // see an inbetween frame with cujo standing, but it kinda makes
  1174.     // it look like he was having a death spasm
  1175.     Cujobody.nextthink = time + 0.01;
  1176.  
  1177.     if (deathframe == 0)
  1178.     {
  1179.       Cujobody.think = CUJO_die10;
  1180.     }
  1181.     else
  1182.     {
  1183.       Cujobody.think = CUJO_dieb10;
  1184.     }
  1185.  
  1186.     if (self.enemy == self.owner)
  1187.     {
  1188.       if (self.health < -35)
  1189.       {
  1190.         centerprint (bot_owner, "You just turned Cujo into puppy chow.\n");
  1191.       }
  1192.       else
  1193.       {
  1194.         centerprint (bot_owner, "You buried your dog.\n");
  1195.       }
  1196.     }
  1197.     else
  1198.     {
  1199.       if (self.health < -35)
  1200.       {
  1201.         centerprint (bot_owner, "Cujo is kibbles and bits.\n");
  1202.       }
  1203.       else
  1204.       {
  1205.         centerprint (bot_owner, "Cujo went to dog heaven.\n");
  1206.       }
  1207.     }
  1208.  
  1209.     remove (self);
  1210.   }
  1211.  
  1212.   bot_owner.Cujo_flag = FALSE;
  1213. };
  1214.  
  1215. /*╔═══════════════════════════════════════════════════════════════════════╗
  1216.   ║                                                                       ║
  1217.   ║ Cujo_TeleportPos                                                      ║
  1218.   ║                                                                       ║
  1219.   ║ Determines the best position for Cujo to spawn in                     ║
  1220.   ║ This procedure is to be called from Cujo's routines, not the player's ║
  1221.   ║                                                                       ║
  1222.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1223.  
  1224. vector () CUJO_TeleportPos =
  1225. {
  1226.   local vector  org, temp, right, front;
  1227.  
  1228.   org = self.owner.origin;
  1229.  
  1230.   makevectors (self.owner.angles);
  1231.  
  1232.   // make front equal 50 units in front of the player
  1233.   front = 50 * normalize (v_forward);
  1234.   // and right equal fifty units to the right
  1235.   right = 50 * normalize (v_right);
  1236.  
  1237.  
  1238.   // right of player
  1239.  
  1240.   temp = org + right;
  1241.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1242.  
  1243.   // left of player
  1244.  
  1245.   temp = org - right;
  1246.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1247.  
  1248.   // in front of player
  1249.  
  1250.   temp = org + front;
  1251.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1252.  
  1253.   // behind player
  1254.  
  1255.   temp = org - front;
  1256.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1257.  
  1258. };
  1259.  
  1260.  
  1261. /*╔═══════════════════════════════════════════════════════════════════════╗
  1262.   ║                                                                       ║
  1263.   ║ Cujo_TeleportToOwner                                                  ║
  1264.   ║                                                                       ║
  1265.   ║ Cujo teleports to his owner, the player                               ║
  1266.   ║                                                                       ║
  1267.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1268.  
  1269. void () CUJO_TeleportToOwner =
  1270. {
  1271.   local vector org;
  1272.  
  1273.   // Check to see if there is room to teleport Cujo here, if not, say so and
  1274.   // exit.
  1275.  
  1276.   org = CUJO_TeleportPos ();
  1277.   if (org == '0 0 0')
  1278.   {
  1279.     centerprint (self, "There is no room for Cujo here.\n");
  1280.     return;
  1281.   }
  1282.  
  1283.   sprint (self.owner, "Whoof!\n");
  1284.   sound (self, CHAN_BODY, "dog/dsight.wav", 1, ATTN_NORM);
  1285.  
  1286.   spawn_tfog (self.origin);
  1287.  
  1288.   self.ideal_yaw = self.angles * '0 1 0';
  1289.   self.pausetime = time + 5;
  1290.   self.nextthink = time + 0.1;
  1291.   self.think = self.Cujo.th_stand;
  1292.  
  1293.   setorigin(self, org);
  1294.  
  1295.   spawn_tfog (self.origin);
  1296.   self.nextthink = time + 0.1;
  1297.   self.think = self.th_stand;
  1298.  
  1299.   return;
  1300. };
  1301.  
  1302. /*╔═══════════════════════════════════════════════════════════════════════╗
  1303.   ║                                                                       ║
  1304.   ║ CUJO_EatGib                                                           ║
  1305.   ║                                                                       ║
  1306.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1307.  
  1308. // this routine is not being used, because Quake was not always reporting
  1309. // a touch when it should have
  1310.  
  1311. // called from player routines, other = gib
  1312.  
  1313. void () CUJO_EatGib =
  1314. {
  1315.   bprint ("-- Entered CUJO_EatGib --\n");
  1316.   bprint (other.classname);
  1317.   bprint (" touched a gib.\n");
  1318.  
  1319.   if ((other.classname == "cujo") && (other.health > 0) && (other.health < 191))
  1320.   {
  1321.     other.health = other.health + 10;
  1322.     other.goalentity = world;
  1323.     other.enemy = world;
  1324.     other.movetarget = world;
  1325.  
  1326.     bprint ("Cujo ate a gib!\n");
  1327.  
  1328.     remove (self);
  1329.   }
  1330.   bprint ("-- Left CUJO_EatGib --\n");
  1331. };
  1332.  
  1333. /*╔═══════════════════════════════════════════════════════════════════════╗
  1334.   ║                                                                       ║
  1335.   ║ CUJO_EatHead                                                          ║
  1336.   ║                                                                       ║
  1337.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1338.  
  1339. // this routine is not being used, because Quake was not always reporting
  1340. // a touch when it should have
  1341.  
  1342. void () CUJO_EatHead =
  1343. {
  1344.   bprint ("-- Entered CUJO_EatHead --\n");
  1345.   bprint (other.classname);
  1346.   bprint (" touched a gib.\n");
  1347.  
  1348.   if ((other.classname == "cujo") && (other.health > 0) && (other.health < 176))
  1349.   {
  1350.     other.health = other.health + 25;
  1351.  
  1352.     other.goalentity = world;
  1353.     other.enemy = world;
  1354.     other.movetarget = world;
  1355.  
  1356.     bprint ("Cujo ate a head!\n");
  1357.  
  1358.     remove (self);
  1359.   }
  1360.   bprint ("-- Left CUJO_EatHead --\n");
  1361. };
  1362.  
  1363. /*╔═══════════════════════════════════════════════════════════════════════╗
  1364.   ║                                                                       ║
  1365.   ║                 Routines called by player, self=player                ║
  1366.   ║                                                                       ║
  1367.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1368.  
  1369. /*╔═══════════════════════════════════════════════════════════════════════╗
  1370.   ║                                                                       ║
  1371.   ║ Cujo_Precache                                                         ║
  1372.   ║                                                                       ║
  1373.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1374.  
  1375. void () CUJO_Precache =
  1376. {
  1377.   // Precache Cujo's sounds
  1378.   precache_model2 ("progs/h_cujo.mdl");
  1379.   precache_model2 ("progs/cujo.mdl");
  1380.  
  1381.   precache_sound2 ("dog/dattack1.wav");
  1382.   precache_sound2 ("dog/ddeath.wav");
  1383.   precache_sound2 ("dog/dpain1.wav");
  1384.   precache_sound2 ("dog/dsight.wav");
  1385.   precache_sound2 ("dog/idle.wav");
  1386. };
  1387.  
  1388. /*╔═══════════════════════════════════════════════════════════════════════╗
  1389.   ║                                                                       ║
  1390.   ║ Cujo_CheckSpawnPos                                                    ║
  1391.   ║                                                                       ║
  1392.   ║ Returns true if Cujo's bounding box area around SpawnPos is free of   ║
  1393.   ║ solid objects                                                         ║
  1394.   ║                                                                       ║
  1395.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1396.  
  1397. float (vector SpawnPos) CUJO_CheckSpawnPos =
  1398. {
  1399.   local  vector  XVector, YVector, ZPosVec, ZNegVec;
  1400.   local  vector  HalfYVec, HalfXVec, HalfZPosVec, HalfZNegVec;
  1401.   local  float   pc;
  1402.  
  1403.   // These vector offsets match Cujo's current size, but do not match
  1404.   // the exact size of the Dog model (which was too big to follow the
  1405.   // player some places), so the dog may at times look like his head
  1406.   // is in a wall, but he shouldn't get stuck (we hope).
  1407.  
  1408.   XVector = '16 0 0';
  1409.   HalfXVec = '8 0 0';
  1410.  
  1411.   YVector = '0 16 0';
  1412.   HalfYVec = '0 16 0';
  1413.  
  1414.   ZPosVec = '0 0 16';
  1415.   HalfZPosVec = '0 0 8';
  1416.   ZNegVec = '0 0 -24';
  1417.   HalfZNegVec = '0 0 -12';
  1418.  
  1419.   // check the very center of the box, just in case something is floating
  1420.   // there (for example, the moving "key cubes" at the end of e1m2)
  1421.  
  1422.   pc = pointcontents (SpawnPos);
  1423.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1424.  
  1425.  
  1426.   // check half the distance to each face of the box.  Again, this is
  1427.   // for floating or thin objects which might fit in between the face
  1428.   // edge and the center of the face
  1429.  
  1430.   pc = pointcontents (SpawnPos + HalfXVec);
  1431.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1432.  
  1433.   pc = pointcontents (SpawnPos - HalfXVec);
  1434.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1435.  
  1436.   pc = pointcontents (SpawnPos + HalfYVec);
  1437.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1438.  
  1439.   pc = pointcontents (SpawnPos - HalfYVec);
  1440.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1441.  
  1442.   pc = pointcontents (SpawnPos + HalfZPosVec);
  1443.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1444.  
  1445.   pc = pointcontents (SpawnPos + HalfZNegVec);
  1446.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1447.  
  1448.  
  1449.   // this checks the six faces of each side of Cujo's bounding box
  1450.  
  1451.   pc = pointcontents (SpawnPos + XVector);
  1452.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1453.  
  1454.   pc = pointcontents (SpawnPos - XVector);
  1455.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1456.  
  1457.   pc = pointcontents (SpawnPos + YVector);
  1458.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1459.  
  1460.   pc = pointcontents (SpawnPos - YVector);
  1461.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1462.  
  1463.   pc = pointcontents (SpawnPos + ZPosVec);
  1464.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1465.  
  1466.   pc = pointcontents (SpawnPos + ZNegVec);
  1467.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1468.  
  1469.  
  1470.   // Check the eight vertices of Cujo's bounding box
  1471.  
  1472.   pc = pointcontents (SpawnPos + XVector + YVector + ZPosVec);
  1473.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1474.  
  1475.   pc = pointcontents (SpawnPos + XVector + YVector + ZNegVec);
  1476.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1477.  
  1478.   pc = pointcontents (SpawnPos + XVector - YVector + ZPosVec);
  1479.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1480.  
  1481.   pc = pointcontents (SpawnPos + XVector - YVector + ZNegVec);
  1482.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1483.  
  1484.   pc = pointcontents (SpawnPos - XVector + YVector + ZPosVec);
  1485.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1486.  
  1487.   pc = pointcontents (SpawnPos - XVector + YVector + ZNegVec);
  1488.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1489.  
  1490.   pc = pointcontents (SpawnPos - XVector - YVector + ZPosVec);
  1491.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1492.  
  1493.   pc = pointcontents (SpawnPos - XVector - YVector + ZNegVec);
  1494.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1495.  
  1496.  
  1497.   // check the center of each edge of the box for content type
  1498.   // first check the center of the top 4 edges of the bounding box
  1499.  
  1500.   pc = pointcontents (SpawnPos + (YVector + ZPosVec));
  1501.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1502.  
  1503.   pc = pointcontents (SpawnPos - (YVector + ZPosVec));
  1504.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1505.  
  1506.   pc = pointcontents (SpawnPos + (XVector + ZPosVec));
  1507.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1508.  
  1509.   pc = pointcontents (SpawnPos - (XVector + ZPosVec));
  1510.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1511.  
  1512.   // now check the center of the middle 4 edges of the bounding box
  1513.  
  1514.   pc = pointcontents (SpawnPos - (XVector + YVector));
  1515.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1516.  
  1517.   pc = pointcontents (SpawnPos - (XVector - YVector));
  1518.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1519.  
  1520.   pc = pointcontents (SpawnPos + (XVector + YVector));
  1521.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1522.  
  1523.   pc = pointcontents (SpawnPos + (XVector - YVector));
  1524.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1525.  
  1526.   // now check the center of the bottom 4 edges of the bounding box
  1527.  
  1528.   pc = pointcontents (SpawnPos + (YVector + ZNegVec));
  1529.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1530.  
  1531.   pc = pointcontents (SpawnPos - (YVector + ZNegVec));
  1532.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1533.  
  1534.   pc = pointcontents (SpawnPos + (XVector + ZNegVec));
  1535.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1536.  
  1537.   pc = pointcontents (SpawnPos - (XVector + ZNegVec));
  1538.   if (pc != CONTENT_EMPTY && pc != CONTENT_SKY) return FALSE;
  1539.  
  1540.   // if that doesn't check enough spots, then that's just too bad.
  1541.  
  1542.   return TRUE;
  1543. };
  1544.  
  1545. /*╔═══════════════════════════════════════════════════════════════════════╗
  1546.   ║                                                                       ║
  1547.   ║ Cujo_SpawnPos                                                         ║
  1548.   ║                                                                       ║
  1549.   ║ Determines the best position for Cujo to spawn in                     ║
  1550.   ║                                                                       ║
  1551.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1552.  
  1553. vector () CUJO_SpawnPos =
  1554. {
  1555.   local vector  org, temp, right, front;
  1556.  
  1557.   org = self.origin;
  1558.  
  1559.   makevectors (self.angles);
  1560.  
  1561.   // make front equal 50 units in front of the player
  1562.   front = 50 * normalize (v_forward);
  1563.   // and right equal fifty units to the right
  1564.   right = 50 * normalize (v_right);
  1565.  
  1566.  
  1567.   // right of player
  1568.  
  1569.   temp = org + right;
  1570.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1571.  
  1572.   // left of player
  1573.  
  1574.   temp = org - right;
  1575.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1576.  
  1577.   // in front of player
  1578.  
  1579.   temp = org + front;
  1580.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1581.  
  1582.   // behind player
  1583.  
  1584.   temp = org - front;
  1585.   if (CUJO_CheckSpawnPos (temp)) return temp;
  1586.  
  1587. };
  1588.  
  1589.  
  1590. /*╔═══════════════════════════════════════════════════════════════════════╗
  1591.   ║                                                                       ║
  1592.   ║ Cujo_Activate                                                         ║
  1593.   ║                                                                       ║
  1594.   ║ Called by player, self = player                                       ║
  1595.   ║                                                                       ║
  1596.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1597.  
  1598. void () CUJO_Activate =
  1599. {
  1600.   local entity  dogbot;
  1601.   local vector  org;
  1602.   local float   bit;
  1603.  
  1604.   // Check to see if there is room to spawn Cujo here, if not, say so and
  1605.   // exit.
  1606.  
  1607.   org = CUJO_SpawnPos ();
  1608.   if (org == '0 0 0')
  1609.   {
  1610.     centerprint (self, "There is no room for Cujo here.\n");
  1611.     return;
  1612.   }
  1613.  
  1614.   // Spawn Cujo
  1615.  
  1616.   dogbot = spawn();
  1617.   dogbot.solid = SOLID_SLIDEBOX;
  1618.   dogbot.movetype = MOVETYPE_STEP;
  1619.  
  1620.   dogbot.angles = self.angles;
  1621.   dogbot.classname = "cujo";
  1622.   dogbot.owner=self;
  1623.   self.Cujo=dogbot;
  1624.   self.Cujo_flag = TRUE;
  1625.   self.Cujo_auto = TRUE;
  1626.   self.Cujo_view = FALSE;
  1627.   self.Cujo.Cujo_attack = TRUE;
  1628.   self.Cujo.Cujo_stay = FALSE;
  1629.  
  1630.   dogbot.takedamage = DAMAGE_AIM;
  1631.   dogbot.goalentity = self;
  1632.   dogbot.movetarget = self;
  1633.   dogbot.pausetime = time + 5;
  1634.   dogbot.ideal_yaw = dogbot.angles * '0 1 0';
  1635.   dogbot.yaw_speed = 30;
  1636.  
  1637.   // lowered Cujo's view offset to 15 above the center of it's bounding box
  1638.   // (I have no idea if this is exactly right...)
  1639.  
  1640.   dogbot.view_ofs = '0 0 15';
  1641.  
  1642.   // this dog has some badass armor
  1643.  
  1644.   if (!deathmatch)
  1645.   {
  1646.     dogbot.health = 200;
  1647.     bit = IT_ARMOR3;
  1648.     dogbot.armortype = 0.8;
  1649.     dogbot.armorvalue = 200;
  1650.     dogbot.items = other.items - (other.items & (IT_ARMOR1 | IT_ARMOR2 | IT_ARMOR3)) + bit;
  1651.   }
  1652.   else
  1653.   {
  1654.     dogbot.health = 100;
  1655.     bit = IT_ARMOR2;
  1656.     dogbot.armortype = 0.6;
  1657.     dogbot.armorvalue = 150;
  1658.     dogbot.items = other.items - (other.items & (IT_ARMOR1 | IT_ARMOR2 | IT_ARMOR3)) + bit;
  1659.   }
  1660.  
  1661.   // Set to automatic mode sequences at startup
  1662.  
  1663.   dogbot.th_stand = CUJO_stand1;
  1664.   dogbot.th_walk = CUJO_walk1;
  1665.   dogbot.th_run = CUJO_run1;
  1666.   dogbot.th_pain = CUJO_pain;
  1667.   dogbot.th_die = CUJO_die;
  1668.   dogbot.th_melee = CUJO_atta1;
  1669.   dogbot.th_missile = CUJO_leap1;
  1670.   dogbot.team = self.team;
  1671.  
  1672.   setmodel (dogbot, "progs/cujo.mdl");
  1673.  
  1674.   // Cujo's size was made the same as the player, (except for his height)
  1675.   // because he was having trouble getting though some narrow doorways
  1676.   // and halls.
  1677.  
  1678.   setsize (dogbot, '-16 -16 -24', '16 16 16');
  1679.   setorigin(dogbot, org);
  1680.  
  1681.   spawn_tfog (dogbot.origin);
  1682.  
  1683.   sound (self, CHAN_BODY, "dog/dattack1.wav", 1, ATTN_NORM);
  1684.   centerprint(self, "Cujo is here.\n");
  1685.  
  1686.   dogbot.nextthink = time + 0.1;
  1687.   dogbot.think = dogbot.th_stand;
  1688.  
  1689.   // remove powerup status from player
  1690.   self.Cujo_avail = 0;
  1691. };
  1692.  
  1693. /*╔═══════════════════════════════════════════════════════════════════════╗
  1694.   ║                                                                       ║
  1695.   ║ Cujo_Deactivate                                                       ║
  1696.   ║                                                                       ║
  1697.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1698.  
  1699. void () CUJO_Deactivate =
  1700. {
  1701.   spawn_tfog (self.Cujo.origin);
  1702.  
  1703.   self.Cujo.nextthink = time + 0.1;
  1704.   self.Cujo.think = SUB_Remove;
  1705.   self.Cujo_flag = FALSE;
  1706.  
  1707.   centerprint (self, "Cujo went back to his doghouse.\n");
  1708. };
  1709.  
  1710. /*╔═══════════════════════════════════════════════════════════════════════╗
  1711.   ║                                                                       ║
  1712.   ║ Cujo_Toggle                                                           ║
  1713.   ║                                                                       ║
  1714.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1715.  
  1716. // ver 1.2 -  Changed spawning rule for deathmatch
  1717.  
  1718. void () CUJO_Toggle =
  1719. {
  1720.   if (deathmatch)
  1721.   {
  1722.     if (self.Cujo_avail)
  1723.     {
  1724.       if (!(self.Cujo_flag))
  1725.         CUJO_Activate ();
  1726.       else
  1727.         sprint (self, "You can only have one Cujo at a time!\n");
  1728.     }
  1729.     else
  1730.       sprint (self, "Cujo is not available.\n");
  1731.   }
  1732.   else
  1733.   {
  1734.     if (!(self.Cujo_flag))
  1735.       CUJO_Activate();
  1736.     else
  1737.       CUJO_Deactivate();
  1738.   }
  1739. };
  1740.  
  1741. /*╔═══════════════════════════════════════════════════════════════════════╗
  1742.   ║                                                                       ║
  1743.   ║ Cujo_AttackToggle                                                     ║
  1744.   ║                                                                       ║
  1745.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1746.  
  1747. void () CUJO_AttackToggle =
  1748. {
  1749.   if (!(self.Cujo_flag))
  1750.   {
  1751.     CUJO_PrintCujoStatus ();
  1752.  
  1753.     return;
  1754.   }
  1755.  
  1756.   if (self.Cujo.Cujo_attack == TRUE)
  1757.   {
  1758.     self.Cujo.Cujo_attack = FALSE;
  1759.     sprint (self, "Cujo will not attack.\n");
  1760.  
  1761.     // reset Cujo's enemy so he doesn't whine even when the enemy goes
  1762.     // out of view.
  1763.  
  1764.     self.Cujo.enemy = world;
  1765.     self.Cujo.oldenemy = world;
  1766.     self.Cujo.goalentity = self;
  1767.     self.Cujo.movetarget = self;
  1768.  
  1769.     self.Cujo.think = self.Cujo.th_stand;
  1770.     self.Cujo.nextthink = time + 0.1;
  1771.   }
  1772.   else
  1773.   {
  1774.     self.Cujo.Cujo_attack = TRUE;
  1775.     sprint (self, "Cujo wants blood!\n");
  1776.   }
  1777.  
  1778.   return;
  1779. };
  1780.  
  1781. /*╔═══════════════════════════════════════════════════════════════════════╗
  1782.   ║                                                                       ║
  1783.   ║ Cujo_Attack                                                           ║
  1784.   ║                                                                       ║
  1785.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1786. /*
  1787. void () CUJO_Attack =
  1788. {
  1789.   local entity oldself;
  1790.   local vector org;
  1791.  
  1792.   if (!(self.Cujo_flag))
  1793.   {
  1794.     CUJO_PrintCujoStatus ();
  1795.  
  1796.     return;
  1797.   }
  1798.  
  1799.   oldself = self;
  1800.   self = self.Cujo;
  1801.  
  1802.   if (self.owner.Cujo_auto && self.enemy != world && self.enemy.health > 1
  1803.   && !(self.enemy.items & IT_INVISIBILITY) && visible(self.enemy))
  1804.   {
  1805.   }
  1806.   else
  1807.   {
  1808.   }
  1809.  
  1810.   self = oldself;
  1811.  
  1812.   return;
  1813. };
  1814. */
  1815. /*╔═══════════════════════════════════════════════════════════════════════╗
  1816.   ║                                                                       ║
  1817.   ║ Cujo_TeleportHome                                                     ║
  1818.   ║                                                                       ║
  1819.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1820.  
  1821. void () CUJO_TeleportHome =
  1822. {
  1823.   local vector org;
  1824.  
  1825.   if (!(self.Cujo_flag))
  1826.   {
  1827.     CUJO_PrintCujoStatus ();
  1828.  
  1829.     return;
  1830.   }
  1831.  
  1832.   spawn_tfog (self.Cujo.origin);
  1833.   self.Cujo.ideal_yaw = self.angles * '0 1 0';
  1834.   self.Cujo.pausetime = time + 2;
  1835.   self.Cujo.nextthink = time + 0.1;
  1836.   self.Cujo.think = self.Cujo.th_stand;
  1837.  
  1838.   org = self.origin + '0 0 0';
  1839.   setorigin(self.Cujo, org);
  1840.  
  1841.   spawn_tfog (self.Cujo.origin);
  1842.   self.Cujo.nextthink = time + 0.1;
  1843.   self.Cujo.think = self.Cujo.th_stand;
  1844.  
  1845.   return;
  1846. };
  1847.  
  1848. /*╔═══════════════════════════════════════════════════════════════════════╗
  1849.   ║                                                                       ║
  1850.   ║ Cujo_LightToggle                                                      ║
  1851.   ║                                                                       ║
  1852.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1853.  
  1854.  
  1855. // This will always be called from the players code, so self is always
  1856. // expected to be the player
  1857.  
  1858. void () CUJO_LightToggle =
  1859. {
  1860.   local   float   effect;
  1861.   local   float   bitmask;
  1862.  
  1863.   if (self.Cujo_flag)
  1864.   {
  1865.     effect = EF_DIMLIGHT;
  1866.  
  1867.     bitmask = (effect) & self.Cujo.effects;
  1868.  
  1869.     if (bitmask == 0)
  1870.     {
  1871.       self.Cujo.effects = self.Cujo.effects | effect;
  1872.     }
  1873.     else
  1874.     {
  1875.       bitmask = !(bitmask);
  1876.       self.Cujo.effects = (self.Cujo.effects) & bitmask;
  1877.     }
  1878.   }
  1879.   else
  1880.     CUJO_PrintCujoStatus ();
  1881.  
  1882.   return;
  1883. };
  1884.  
  1885. /*╔═══════════════════════════════════════════════════════════════════════╗
  1886.   ║                                                                       ║
  1887.   ║ Cujo_SetDogView                                                       ║
  1888.   ║                                                                       ║
  1889.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1890.  
  1891. // This will always be called from the players code, so self is always
  1892. // expected to be the player
  1893.  
  1894. void () CUJO_SetDogView =
  1895. {
  1896.   if (self.Cujo_flag)
  1897.   {
  1898.     // Set view point
  1899.     msg_entity = self;
  1900.     WriteByte (MSG_ONE, 5);                 // SVC, set the viewport
  1901.     WriteEntity (MSG_ONE, self.Cujo);
  1902.  
  1903.     // set the view angles to Cujo's view angles
  1904. //    WriteByte (MSG_ONE, 10);               // SVC, set view angles
  1905. //    WriteAngle(MSG_ONE, self.Cujo.angles_x);
  1906. //    WriteAngle(MSG_ONE, self.Cujo.angles_y);
  1907. //    WriteAngle(MSG_ONE, self.Cujo.angles_z);
  1908.  
  1909.     self.weaponmodel = "";
  1910.     self.weaponframe = 0;
  1911.  
  1912.     self.Cujo.Cujo_view = TRUE;
  1913.   }
  1914.   else
  1915.     CUJO_PrintCujoStatus ();
  1916.  
  1917.   return;
  1918.  
  1919. };
  1920.  
  1921. /*╔═══════════════════════════════════════════════════════════════════════╗
  1922.   ║                                                                       ║
  1923.   ║ Cujo_SetPlayerView                                                    ║
  1924.   ║                                                                       ║
  1925.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1926.  
  1927.  
  1928. // This will always be called from the players code, so self is always
  1929. // expected to be the player
  1930.  
  1931. void () CUJO_SetPlayerView =
  1932. {
  1933.   if (self.Cujo_flag)
  1934.   {
  1935.     // Set view point
  1936.     msg_entity = self;
  1937.     WriteByte (MSG_ONE, 5);                 // service, set the viewport
  1938.     WriteEntity (MSG_ONE, self);
  1939.  
  1940. //    WriteByte (MSG_ONE, 10);               // SVC, set view angles
  1941. //    WriteAngle(MSG_ONE, self.angles_x);    // tilt
  1942. //    WriteAngle(MSG_ONE, self.angles_y);    // yaw
  1943. //    WriteAngle(MSG_ONE, self.angles_z);    // flip
  1944.  
  1945.     W_SetCurrentAmmo ();
  1946.  
  1947.     self.Cujo.Cujo_view = FALSE;
  1948.  
  1949.   }
  1950.   else
  1951.     CUJO_PrintCujoStatus ();
  1952.  
  1953.   return;
  1954. };
  1955.  
  1956. /*╔═══════════════════════════════════════════════════════════════════════╗
  1957.   ║                                                                       ║
  1958.   ║ Cujo_Stay                                                             ║
  1959.   ║                                                                       ║
  1960.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1961.  
  1962. // always called from player code.  Toggles self.Cujo.Cujo_stay which
  1963. // is used in follow routines to determine if Cujo should stay or follow
  1964.  
  1965. void () CUJO_Stay =
  1966. {
  1967.   if (self.Cujo_flag)
  1968.   {
  1969.     // toggle Cujo's stay flag
  1970.     if (self.Cujo.Cujo_stay == TRUE)
  1971.     {
  1972.       self.Cujo.Cujo_stay = FALSE;
  1973.       sprint (self, "Cujo is following.\n");
  1974.     }
  1975.     else
  1976.     {
  1977.       self.Cujo.Cujo_stay = TRUE;
  1978.       sprint (self, "Cujo is staying.\n");
  1979.     }
  1980.  
  1981.   }
  1982.   else
  1983.     CUJO_PrintCujoStatus ();
  1984.  
  1985.   return;
  1986. };
  1987.  
  1988. /*╔═══════════════════════════════════════════════════════════════════════╗
  1989.   ║                                                                       ║
  1990.   ║ Cujo_GiveStatus                                                       ║
  1991.   ║                                                                       ║
  1992.   ╚═══════════════════════════════════════════════════════════════════════╝*/
  1993.  
  1994. // called from player code, ie. self = player
  1995.  
  1996. void () CUJO_GiveStatus =
  1997. {
  1998.   local   string  str_temp;
  1999.   local   float   ftemp;
  2000.  
  2001.   if (self.Cujo_flag)
  2002.   {
  2003.     if (visible (self.Cujo))
  2004.     {
  2005.       sprint (self, "Cujo's health is ");
  2006.  
  2007.       if (!deathmatch)
  2008.         ftemp = self.Cujo.health / 2;
  2009.       else
  2010.         ftemp = self.Cujo.health;
  2011.  
  2012.       str_temp = ftos (ftemp);
  2013.       sprint (self, str_temp);
  2014.       sprint (self, "%.\n");
  2015.  
  2016.       sprint (self, "His armor is ");
  2017.  
  2018.       if (!deathmatch)
  2019.         ftemp = self.Cujo.armorvalue / 2;
  2020.       else
  2021.         ftemp = self.Cujo.armorvalue / 1.5;
  2022.  
  2023.       str_temp = ftos (ftemp);
  2024.       sprint (self, str_temp);
  2025.       sprint (self, "%.\n");
  2026.     }
  2027.     else
  2028.     {
  2029.       sprint (self, "You must be able to see Cujo to determine his status.\n");
  2030.     }
  2031.   }
  2032.   else
  2033.     CUJO_PrintCujoStatus ();
  2034.  
  2035.   return;
  2036. };