rocket.cc 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. /********************************************************************** <BR>
  2. This file is part of Crack dot Com's free source code release of
  3. Golgotha. <a href="http://www.crack.com/golgotha_release"> <BR> for
  4. information about compiling & licensing issues visit this URL</a>
  5. <PRE> If that doesn't help, contact Jonathan Clark at
  6. golgotha_source@usa.net (Subject should have "GOLG" in it)
  7. ***********************************************************************/
  8. #include "g1_object.hh"
  9. #include "objs/model_draw.hh"
  10. #include "math/num_type.hh"
  11. #include "objs/rocket.hh"
  12. #include "tile.hh"
  13. #include "objs/explosion1.hh"
  14. #include "saver.hh"
  15. #include "objs/fire_angle.hh"
  16. #include "math/pi.hh"
  17. #include "objs/map_piece.hh"
  18. #include "map.hh"
  19. #include "map_man.hh"
  20. #include "math/angle.hh"
  21. #include "math/trig.hh"
  22. #include "image/image.hh"
  23. #include "objs/smoke_trail.hh"
  24. #include "object_definer.hh"
  25. #include "g1_render.hh"
  26. #include "g1_rand.hh"
  27. static g1_object_type shrapnel_type, explosion_type, smoke_trail_type;
  28. void g1_rocket_init()
  29. {
  30. shrapnel_type = g1_get_object_type("shrapnel");
  31. explosion_type = g1_get_object_type("explosion1");
  32. smoke_trail_type = g1_get_object_type("smoke_trail");
  33. }
  34. g1_object_definer<g1_rocket_class>
  35. g1_rocket_def("rocket",
  36. g1_object_definition_class::MOVABLE,
  37. g1_rocket_init);
  38. g1_rocket_class::g1_rocket_class(g1_object_type id,
  39. g1_loader_class *fp)
  40. : g1_object_class(id,fp)
  41. {
  42. draw_params.setup("missile");
  43. if (fp && fp->check_version(G1_ROCKET_DATA_VERSION))
  44. {
  45. zv=fp->read_float();
  46. speed=fp->read_float();
  47. fp->read_reference(who_fired_me);
  48. fp->read_reference(track_object);
  49. i4_warning("loaded rocket that does no damage");
  50. fp->end_version(I4_LF);
  51. }
  52. else
  53. {
  54. zv=0;
  55. speed=0;
  56. }
  57. damage = 0;
  58. radar_type=G1_RADAR_WEAPON;
  59. set_flag(AERIAL |
  60. HIT_GROUND |
  61. SHADOWED, 1);
  62. }
  63. void g1_rocket_class::save(g1_saver_class *fp)
  64. {
  65. // save data associated with base classes
  66. g1_object_class::save(fp);
  67. fp->start_version(G1_ROCKET_DATA_VERSION);
  68. fp->write_float(zv);
  69. fp->write_float(speed);
  70. fp->write_reference(who_fired_me);
  71. fp->write_reference(track_object);
  72. fp->end_version();
  73. }
  74. void g1_rocket_class::fire_at(i4_float sx, i4_float sy, i4_float sz,
  75. i4_float angle,
  76. g1_map_piece_class *kill_target,
  77. g1_map_piece_class *owner,
  78. w32 _damage)
  79. {
  80. damage = _damage;
  81. lx=x=sx; ly=y=sy; lh=h=sz;
  82. i4_float x_y_dist = sqrt((kill_target->x - sx)*(kill_target->x - sx) +
  83. (kill_target->y - sy)*(kill_target->y - sy));
  84. i4_float angle_with_ground;
  85. if (!g1_calc_fire_angle(x_y_dist, sz, kill_target->h, g(), vo(), angle_with_ground))
  86. angle_with_ground=i4_pi()/4;
  87. speed=cos(angle_with_ground)*vo();
  88. zv=sin(angle_with_ground)*vo();
  89. ltheta=theta=angle;
  90. pitch = lpitch = atan(-zv/speed);
  91. player_num=owner->player_num;
  92. track_object = kill_target;
  93. who_fired_me = owner;
  94. occupy_location();
  95. request_think();
  96. }
  97. void g1_rocket_class::fire(i4_float sx, i4_float sy, i4_float sz,
  98. i4_angle angle,
  99. g1_map_piece_class *owner,
  100. w32 _damage)
  101. {
  102. damage = _damage;
  103. i4_float angle_with_ground=i4_pi()/4;
  104. speed=cos(angle_with_ground)*vo();
  105. zv=sin(angle_with_ground)*vo();
  106. theta=angle;
  107. who_fired_me = owner;
  108. player_num=owner->player_num;
  109. occupy_location();
  110. request_think();
  111. }
  112. void i4_normalize_angle(i4_float &angle);
  113. void g1_rocket_class::think()
  114. {
  115. zv-=g();
  116. pitch = atan(-zv/speed);
  117. i4_normalize_angle(pitch);
  118. move(cos(theta) * speed, sin(theta) * speed, zv);
  119. }
  120. void g1_rocket_class::draw(g1_draw_context_class *context)
  121. {
  122. g1_model_draw(this, draw_params, context);
  123. if (smoke_trail.valid())
  124. smoke_trail->draw(context);
  125. }
  126. i4_bool g1_rocket_class::move(i4_float x_amount,
  127. i4_float y_amount,
  128. i4_float z_amount)
  129. {
  130. g1_smoke_trail_class *s;
  131. if (!smoke_trail.valid())
  132. {
  133. s=(g1_smoke_trail_class *)g1_create_object(smoke_trail_type);
  134. if (s)
  135. {
  136. s->setup(x, y, h, 0.01, 0.1, 0xffd000, 0xffffff); // orange to white
  137. s->occupy_location();
  138. smoke_trail=s;
  139. }
  140. }
  141. else s=(g1_smoke_trail_class *)smoke_trail.get();
  142. g1_object_class *hit = NULL;
  143. unoccupy_location();
  144. i4_3d_vector ray(x_amount, y_amount, z_amount);
  145. if (!g1_get_map()->check_non_player_collision(player_num,
  146. i4_3d_vector(x,y,h),
  147. ray,
  148. hit))
  149. {
  150. x += x_amount;
  151. y += y_amount;
  152. h += z_amount;
  153. request_think();
  154. occupy_location();
  155. if (s)
  156. s->update_head(x,y,h);
  157. return i4_T;
  158. }
  159. else
  160. {
  161. if (s) // delete the smoke trail
  162. {
  163. s->unoccupy_location();
  164. s->request_remove();
  165. }
  166. // explode
  167. if (hit)
  168. {
  169. if (fabs(hit->h - h)<0.2)
  170. {
  171. hit->damage(this, damage, i4_3d_vector(x_amount,y_amount,z_amount));
  172. g1_explosion1_class *explosion;
  173. explosion=(g1_explosion1_class *)g1_create_object(explosion_type);
  174. i4_float rx,ry,rh;
  175. rx = g1_float_rand(34) *0.2 - 0.1;
  176. ry = g1_float_rand(10) *0.2 - 0.1;
  177. rh = g1_float_rand(02) *0.2 - 0.1;
  178. explosion->setup(hit->x+rx,hit->y+ry,hit->h + rh);
  179. x = hit->x + rx;
  180. y = hit->y + ry;
  181. h = hit->h + rh;
  182. request_remove();
  183. return i4_F;;
  184. }
  185. x += x_amount;
  186. y += y_amount;
  187. h += z_amount;
  188. request_think();
  189. occupy_location();
  190. return i4_T;
  191. }
  192. else if (who_fired_me.valid())
  193. {
  194. g1_explosion1_class *explosion=(g1_explosion1_class *)g1_create_object(explosion_type);
  195. i4_float rx,ry,rh;
  196. rx = g1_float_rand(34) *0.2 - 0.1;
  197. ry = g1_float_rand(10) *0.2 - 0.1;
  198. rh = g1_float_rand(02) *0.2 - 0.1;
  199. explosion->setup(lx+rx,ly+ry,lh+rh);
  200. }
  201. request_remove();
  202. return i4_F;
  203. }
  204. }