light_o.cc 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  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 "objs/light_o.hh"
  9. #include "saver.hh"
  10. #include "objs/model_draw.hh"
  11. #include "map.hh"
  12. #include "math/num_type.hh"
  13. #include "object_definer.hh"
  14. #include "map_man.hh"
  15. #include "flare.hh"
  16. #include "time/profile.hh"
  17. #include "map_vert.hh"
  18. static i4_profile_class pf_light_occupy("light::occupy"), pf_light_unoccupy("light::unoccupy");
  19. g1_object_type g1_lightbulb_type;
  20. void g1_light_object_init();
  21. g1_object_definer<g1_light_object_class>
  22. g1_light_object_def("lightbulb",
  23. g1_object_definition_class::EDITOR_SELECTABLE,
  24. g1_light_object_init);
  25. void g1_light_object_init()
  26. {
  27. g1_lightbulb_type = g1_light_object_def.type;
  28. }
  29. void g1_light_object_class::setup(float _x, float _y, float _h,
  30. float _r, float _g, float _b, float _white,
  31. float min_light_contribute,
  32. float linear_contribute,
  33. float geometric_contribute)
  34. {
  35. r=_r;
  36. g=_g;
  37. b=_b;
  38. x=lx=_x;
  39. y=ly=_y;
  40. h=lh=_h;
  41. int occ=get_flag(MAP_OCCUPIED);
  42. if (occ)
  43. unoccupy_location();
  44. if (add_intensities)
  45. i4_free(add_intensities);
  46. c1=min_light_contribute;
  47. c2=linear_contribute;
  48. c3=geometric_contribute;
  49. if (c3<0.0016) // maximum radius is 25 squares for now
  50. c3=0.0016;
  51. change_radius=i4_f_to_i(sqrt((32-c1)/c3));
  52. sw32 w=change_radius*2+1,h=change_radius*2+1;
  53. add_intensities=(w32 *)i4_malloc(w*h*sizeof(w32),
  54. "light map restore");
  55. if (occ)
  56. occupy_location();
  57. }
  58. g1_light_object_class::~g1_light_object_class()
  59. {
  60. if (add_intensities)
  61. i4_free(add_intensities);
  62. }
  63. g1_light_object_class::g1_light_object_class(g1_object_type id,
  64. g1_loader_class *fp)
  65. : g1_object_class(id,fp)
  66. {
  67. add_intensities=0;
  68. if (fp && fp->check_version(DATA_VERSION))
  69. {
  70. float _r, _g, _b, _white, _c1, _c2, _c3;
  71. fp->read_format("fffffff", &_r, &_g, &_b, &_white, &_c1, &_c2, &_c3);
  72. setup(x,y,h, _r, _g, _b, _white, _c1, _c2, _c3);
  73. fp->end_version(I4_LF);
  74. }
  75. else
  76. {
  77. r=g=b=1;
  78. white=1;
  79. c1=0.1; c2=0.25; c3=0.5;
  80. h=2;
  81. }
  82. draw_params.setup("lightbulb");
  83. }
  84. void g1_light_object_class::save(g1_saver_class *fp)
  85. {
  86. g1_object_class::save(fp);
  87. fp->start_version(DATA_VERSION);
  88. fp->write_format("fffffff", &r,&g,&b, &white, &c1,&c2,&c3);
  89. fp->end_version();
  90. }
  91. void g1_light_object_class::draw(g1_draw_context_class *context)
  92. {
  93. g1_editor_model_draw(this, draw_params, context);
  94. }
  95. void g1_light_object_class::move(float nx, float ny, float nh)
  96. {
  97. unoccupy_location();
  98. lx=x; ly=y; lh=h;
  99. x=nx; y=ny; h=nh;
  100. occupy_location();
  101. }
  102. void g1_light_object_class::think()
  103. {
  104. }
  105. i4_bool g1_light_object_class::occupy_location()
  106. {
  107. if (!add_intensities)
  108. {
  109. i4_warning("call light::setup before occupy_location");
  110. return i4_F;
  111. }
  112. pf_light_occupy.start();
  113. if (g1_object_class::occupy_location())
  114. {
  115. sw32 ix=i4_f_to_i(x), iy=i4_f_to_i(y);
  116. w32 *a=add_intensities;
  117. for (int ty=-change_radius+iy; ty<=change_radius+iy; ty++)
  118. {
  119. sw32 start_x=ix-change_radius;
  120. if (start_x<0) start_x=0;
  121. g1_map_vertex_class *v=g1_get_map()->vertex(start_x,ty);
  122. for (int tx=-change_radius+ix; tx<=change_radius+ix; tx++, a++)
  123. {
  124. if (tx>=0 && ty>=0 && tx<=g1_get_map()->width() && ty<=g1_get_map()->height())
  125. {
  126. i4_3d_vector normal;
  127. i4_float tz;
  128. w32 old_rgb=v->dynamic_light;
  129. v->get_normal(normal, tx, ty);
  130. tz=v->get_height();
  131. i4_3d_vector dir=i4_3d_vector(x-tx,
  132. y-ty,
  133. h-tz);
  134. i4_float dist=sqrt(dir.x*dir.x + dir.y*dir.y + dir.z*dir.z);
  135. i4_float odist=1.0/dist;
  136. dir.x*=odist; // normalize the light direction vector
  137. dir.y*=odist;
  138. dir.z*=odist;
  139. i4_float ndl = normal.dot(dir);
  140. i4_float atten = 1.0/(c1 + c3*dist*dist);
  141. if (atten>1) atten=1;
  142. i4_float intensity=ndl*atten;
  143. if (intensity<0) intensity=0;
  144. sw32 ra,ga,ba, or,og,ob;
  145. ra=i4_f_to_i(intensity * r * 255); // calculate how much to add to the current light
  146. ga=i4_f_to_i(intensity * g * 255);
  147. ba=i4_f_to_i(intensity * b * 255);
  148. or=(old_rgb>>16)&255; // grab the old light values
  149. og=(old_rgb>>8)&255;
  150. ob=(old_rgb)&255;
  151. if (ra+or>255) ra=255-or; // adjust for overflow
  152. if (ga+og>255) ga=255-og;
  153. if (ba+ob>255) ba=255-ob;
  154. *a=(ra<<16)|(ga<<8)|ba; // store the added amount so we can subtract out later
  155. v->dynamic_light=((ra+or)<<16) | ((ga+og)<<8) | (ba+ob);
  156. v->light_sum|=0x80000000;
  157. v++;
  158. }
  159. }
  160. }
  161. pf_light_occupy.stop();
  162. return i4_T;
  163. }
  164. else
  165. {
  166. pf_light_occupy.stop();
  167. return i4_F;
  168. }
  169. }
  170. void g1_light_object_class::unoccupy_location()
  171. {
  172. if (!add_intensities)
  173. return ;
  174. pf_light_unoccupy.start();
  175. g1_object_class::unoccupy_location();
  176. sw32 w=change_radius*2+1,h=change_radius*2+1;
  177. sw32 ix=i4_f_to_i(x), iy=i4_f_to_i(y);
  178. w32 *a=add_intensities;
  179. for (int ty=-change_radius+iy; ty<=change_radius+iy; ty++)
  180. {
  181. sw32 start_x=ix-change_radius;
  182. if (start_x<0) start_x=0;
  183. g1_map_vertex_class *v=g1_get_map()->vertex(start_x,ty);
  184. for (int tx=-change_radius+ix; tx<=change_radius+ix; tx++, a++)
  185. {
  186. if (tx>=0 && ty>=0 && tx<=g1_get_map()->width() && ty<=g1_get_map()->height())
  187. {
  188. v->dynamic_light-=*a;
  189. v++;
  190. v->light_sum=0x80000000;
  191. }
  192. }
  193. }
  194. pf_light_unoccupy.stop();
  195. }