bolt.cc 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  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 "math/pi.hh"
  9. #include "math/trig.hh"
  10. #include "math/angle.hh"
  11. #include "g1_rand.hh"
  12. #include "saver.hh"
  13. #include "map_cell.hh"
  14. #include "map.hh"
  15. #include "map_man.hh"
  16. #include "g1_render.hh"
  17. #include "object_definer.hh"
  18. #include "objs/light_o.hh"
  19. #include "objs/bolt.hh"
  20. #include "time/profile.hh"
  21. #include "camera.hh"
  22. #include "resources.hh"
  23. #include "lisp/lisp.hh"
  24. static i4_profile_class pf_bolt("bolt_think");
  25. S1_SFX(fire_sfx, "fire/electric_tower_firing_three_22khz_lp.wav", S1_3D, 30);
  26. static li_symbol_ref light_type("lightbulb");
  27. g1_object_definer<g1_bolt_class>
  28. g1_bolt_def("bolt");
  29. g1_bolt_class::g1_bolt_class(g1_object_type id, g1_loader_class *fp)
  30. : g1_object_class(id,fp)
  31. //{{{
  32. {
  33. memset(arc,0,sizeof(arc_point) * NUM_ARCS);
  34. first = 1;
  35. size = 0.8;
  36. w1 = w2 = 0.02;
  37. a1 = a2 = 1.0;
  38. c1 = 0xffffff;
  39. c2 = 0x8800ff;
  40. lit = i4_F;
  41. }
  42. //}}}
  43. void g1_bolt_class::setup(const i4_3d_vector &start_pos,
  44. const i4_3d_vector &_target_pos,
  45. g1_object_class *_originator,
  46. g1_object_class *_target)
  47. //{{{
  48. {
  49. move(start_pos);
  50. ticks = 1;
  51. originator = _originator;
  52. target = _target;
  53. if (target.valid())
  54. target_pos.set(target->x,target->y,target->h);
  55. else
  56. target_pos = _target_pos;
  57. if (!get_flag(MAP_OCCUPIED))
  58. {
  59. occupy_location();
  60. request_think();
  61. }
  62. if (lit && get_flag(MAP_OCCUPIED))
  63. create_light();
  64. }
  65. //}}}
  66. g1_bolt_class::~g1_bolt_class()
  67. //{{{
  68. {
  69. }
  70. //}}}
  71. void g1_bolt_class::save(g1_saver_class *fp)
  72. //{{{
  73. {
  74. g1_object_class::save(fp);
  75. }
  76. //}}}
  77. void g1_bolt_class::draw(g1_draw_context_class *context)
  78. //{{{
  79. {
  80. int i;
  81. i4_3d_point_class pts[g1_bolt_class::NUM_ARCS];
  82. for (i=0;i<g1_bolt_class::NUM_ARCS;i++)
  83. pts[i].interpolate(arc[i].lposition,arc[i].position,g1_render.frame_ratio);
  84. g1_render.add_translucent_trail(context->transform,pts,
  85. g1_bolt_class::NUM_ARCS,
  86. w1,w2,a1,a2,c1,c2);
  87. }
  88. //}}}
  89. void g1_bolt_class::copy_old_points()
  90. //{{{
  91. {
  92. sw32 i;
  93. for (i=0;i<NUM_ARCS;i++)
  94. arc[i].lposition = arc[i].position;
  95. }
  96. //}}}
  97. void g1_bolt_class::arc_to(const i4_3d_vector &target)
  98. //{{{
  99. {
  100. i4_3d_vector r,p;
  101. i4_float map_point_height;
  102. i4_3d_vector point, ray;
  103. point.set(x,y,h);
  104. ray = target;
  105. ray -= point;
  106. ray /= NUM_ARCS-1;
  107. sw32 i;
  108. for (i=0; i<NUM_ARCS; i++)
  109. {
  110. i4_float sin_factor = sin(i4_pi()*i/(NUM_ARCS-1));
  111. r.x = ((g1_rand(23)&0xFFFF)/((i4_float)0xffff) - 0.5) * size*sin_factor;
  112. r.y = ((g1_rand(32)&0xFFFF)/((i4_float)0xffff) - 0.5) * size*sin_factor;
  113. r.z = ((g1_rand(45)&0xFFFF)/((i4_float)0xffff) - 0.5) * size*sin_factor;
  114. p.set(point.x + ray.x*i + r.x,
  115. point.y + ray.y*i + r.y,
  116. point.z + ray.z*i + r.z);
  117. map_point_height = g1_get_map()->map_height(p.x,p.y,p.z) + 0.05;
  118. if (p.z < map_point_height)
  119. p.z = map_point_height;
  120. arc[i].position = p;
  121. }
  122. if (end_light.get())
  123. end_light->move(target.x, target.y, target.z);
  124. }
  125. //}}}
  126. void g1_bolt_class::think()
  127. //{{{
  128. {
  129. pf_bolt.start();
  130. if (target.valid())
  131. g1_apply_damage(this, originator.get(), target.get(), i4_3d_vector(0,0,0));
  132. pf_bolt.stop();
  133. }
  134. //}}}
  135. void g1_bolt_class::request_remove()
  136. {
  137. s1_end_looping(sfx_loop);
  138. }
  139. void g1_bolt_class::post_think()
  140. //{{{
  141. {
  142. if (ticks<=0)
  143. {
  144. unoccupy_location();
  145. destroy_light();
  146. request_remove();
  147. return ;
  148. }
  149. copy_old_points();
  150. if (target.valid())
  151. target_pos.set(target->x,target->y,target->h);
  152. arc_to(target_pos);
  153. if (first)
  154. copy_old_points();
  155. first=0;
  156. ticks--;
  157. request_think();
  158. }
  159. //}}}
  160. void g1_bolt_class::move(const i4_3d_vector &pos)
  161. //{{{
  162. {
  163. x = pos.x;
  164. y = pos.y;
  165. h = pos.z;
  166. if (light.valid())
  167. light->move(x,y,h);
  168. }
  169. //}}}
  170. void g1_bolt_class::create_light()
  171. {
  172. sfx_loop.setup(x,y,h,0,0,0,1);
  173. fire_sfx.play_looping(sfx_loop);
  174. if (!light.get())
  175. {
  176. float r=g1_resources.visual_radius();
  177. if (g1_current_view_state()->dist_sqrd(i4_3d_vector(x,y,h))<r*r)
  178. {
  179. light = (g1_light_object_class *)g1_create_object(g1_get_object_type(light_type.get()));
  180. light->setup(x,y,h+0.3,
  181. i4_float(c1>>16&0xff)/256.0,
  182. i4_float(c1>>8&0xff)/256.0,
  183. i4_float(c1&0xff)/256.0,
  184. 1);
  185. light->occupy_location();
  186. end_light = (g1_light_object_class *)g1_create_object(g1_get_object_type(light_type.get()));
  187. end_light->setup(target_pos.x, target_pos.y, target_pos.z,
  188. i4_float(c2>>16&0xff)/256.0,
  189. i4_float(c2>>8&0xff)/256.0,
  190. i4_float(c2&0xff)/256.0,
  191. 1);
  192. end_light->occupy_location();
  193. }
  194. }
  195. }
  196. void g1_bolt_class::destroy_light()
  197. {
  198. s1_end_looping(sfx_loop);
  199. if (light.get())
  200. {
  201. light->unoccupy_location();
  202. light->request_remove();
  203. light=0;
  204. }
  205. if (end_light.get())
  206. {
  207. end_light->unoccupy_location();
  208. end_light->request_remove();
  209. end_light=0;
  210. }
  211. }
  212. //{{{ Emacs Locals
  213. // Local Variables:
  214. // folded-file: t
  215. // End:
  216. //}}}