map_vert.hh 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  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. #ifndef G1_MAP_VERTEX_HH
  9. #define G1_MAP_VERTEX_HH
  10. #include "arch.hh"
  11. #include "f_tables.hh"
  12. #include "math/vector.hh"
  13. #include "math/transform.hh"
  14. #include "r1_clip.hh"
  15. #include "r1_api.hh"
  16. #include "map_man.hh"
  17. class i4_file_class;
  18. class i4_saver_class;
  19. extern i4_float g1_vert_height_table[256];
  20. class g1_map_vertex_class
  21. {
  22. public:
  23. i4_3d_vector v; // view space transformed coordinates
  24. i4_float px, py; // projected screen x & y
  25. i4_float w; // 1/z
  26. // 8-8-8 lighting values for r,g,b, dynamic light cannot be recalculated
  27. // dynamic light is assummed to come from straight down, this value is changed by light objects
  28. // but is normally 0
  29. w32 dynamic_light;
  30. // sum of dynamic, static, and global light values packed into 8-8-8, top bit indicates
  31. // need to recalculate. Set need-to-recalc if dynamic light or normal changes
  32. w32 light_sum;
  33. w16 normal; // 5-5-5, x,y,z, top bit indicates needs recalc
  34. w16 flags;
  35. // amount of light (0..255) visible from the directional light (in the direction of the
  36. // of the global directional light) - cannot be calculated in game because it requires
  37. // ray-tracing and radiocity (not implemented yet)
  38. w8 static_intensity;
  39. // 0-256 shadow subtraction for clouds
  40. w8 shadow_subtract;
  41. w8 height; // height above ground increments of .05
  42. w8 clip_code;
  43. i4_float t_height; // height with t-intersection adjustment
  44. enum {
  45. SELECTED = (1<<0), // only used by editor
  46. FOGGED = (1<<1), // fog of war
  47. TRANSFORMED = (1<<2),
  48. PROJECTED = (1<<3),
  49. CLIP_CODE_CALCULATED= (1<<4),
  50. W_CALCULATED = (1<<5),
  51. NEED_UNDO_SAVE = (1<<6), // only used by editor
  52. WAS_DRAWN_LAST_FRAME= (1<<7), // only used by editor
  53. APPLY_WAVE_FUNCTION = (1<<8), // if vert is part of water
  54. T_INTERSECTION = (1<<9), // used to determine that this is a T intersection
  55. };
  56. enum {SAVED_FLAGS = SELECTED | FOGGED};
  57. w16 get_flag(w16 f) { return (flags & f); }
  58. void set_flag(w16 f, int on_off)
  59. {
  60. if (on_off) flags|=f;
  61. else flags&=~f;
  62. }
  63. w8 is_transformed() { return get_flag(TRANSFORMED); }
  64. void set_is_transformed(w8 yes_no) { set_flag(TRANSFORMED, yes_no); }
  65. w8 is_projected() { return get_flag(PROJECTED); }
  66. void set_is_projected(w8 yes_no) { set_flag(PROJECTED, yes_no); }
  67. w8 is_selected() { return get_flag(SELECTED); }
  68. void set_is_selected(w8 yes_no) { set_flag(SELECTED, yes_no); }
  69. w8 need_undo() { return get_flag(NEED_UNDO_SAVE); }
  70. void set_need_undo(w8 yes_no) { set_flag(NEED_UNDO_SAVE, yes_no); }
  71. w8 is_clipped() { return get_flag(CLIP_CODE_CALCULATED); }
  72. void set_is_clipped(w8 yes_no) { set_flag(CLIP_CODE_CALCULATED, yes_no); }
  73. w8 is_w_calculated() { return get_flag(W_CALCULATED); }
  74. void set_is_w_calculated(w8 yes_no) { set_flag(W_CALCULATED, yes_no); }
  75. void clear_calculations()
  76. {
  77. flags &= ~(PROJECTED | TRANSFORMED | CLIP_CODE_CALCULATED | W_CALCULATED | T_INTERSECTION);
  78. }
  79. float get_non_dynamic_ligth_intensity(int cvx, int cvy);
  80. void recalc_normal(int cvx, int cvy);
  81. void recalc_light_sum(int cvx, int cvy);
  82. void get_normal(i4_3d_vector &v, int cvx, int cvy)
  83. {
  84. if (normal & 0x8000)
  85. recalc_normal(cvx, cvy);
  86. v.x=g1_table_0_31_to_n1_1[(normal>>10)&31];
  87. v.y=g1_table_0_31_to_n1_1[(normal>>5)&31];
  88. v.z=g1_table_0_31_to_n1_1[(normal)&31];
  89. }
  90. void get_rgb(i4_float &r, i4_float &g, i4_float &b, int cvx, int cvy)
  91. {
  92. if (light_sum & 0x80000000)
  93. recalc_light_sum(cvx, cvy);
  94. w32 ls=light_sum;
  95. r=g1_table_0_255_to_0_1[((ls>>16)&0xff)];
  96. g=g1_table_0_255_to_0_1[((ls>>8)&0xff)];
  97. b=g1_table_0_255_to_0_1[((ls)&0xff)];
  98. }
  99. float get_r(int cvx, int cvy)
  100. {
  101. if (light_sum & 0x80000000)
  102. recalc_light_sum(cvx, cvy);
  103. return g1_table_0_255_to_0_1[((light_sum>>16)&0xff)];
  104. }
  105. i4_float get_height() { return g1_vert_height_table[height]; }
  106. void wave_transform(i4_transform_class &t, float map_x, float map_y);
  107. void transform(i4_transform_class &t, int map_x, int map_y,float &xscale, float &yscale)
  108. {
  109. if (!is_transformed())
  110. {
  111. if (flags & APPLY_WAVE_FUNCTION)
  112. wave_transform(t, map_x, map_y);
  113. else
  114. t.transform(i4_3d_point_class(map_x, map_y, t_height), v);
  115. v.x *= xscale;
  116. v.y *= yscale;
  117. set_is_transformed(i4_T);
  118. }
  119. }
  120. void calculate_w()
  121. {
  122. w=r1_ooz(v.z);
  123. set_is_w_calculated(i4_T);
  124. }
  125. void project(i4_float win_center_x, i4_float win_center_y)
  126. {
  127. if (!is_projected())
  128. {
  129. if (!is_w_calculated())
  130. calculate_w();
  131. px=v.x * w * win_center_x + win_center_x;
  132. py=v.y * w * win_center_y + win_center_y;
  133. set_is_projected(i4_T);
  134. }
  135. }
  136. w8 calc_clip_code()
  137. {
  138. if (!is_clipped())
  139. {
  140. clip_code = r1_calc_outcode(v);
  141. set_is_clipped(1);
  142. }
  143. return clip_code;
  144. }
  145. void set_r1_vert(r1_vert *r)
  146. {
  147. r->v.x = v.x;
  148. r->v.y = v.y;
  149. r->v.z = v.z;
  150. r->px = px;
  151. r->py = py;
  152. r->w = w;
  153. r->outcode = clip_code;
  154. }
  155. void load_v1(i4_file_class *fp);
  156. void load_v2(i4_file_class *fp);
  157. void load_v4(i4_file_class *fp);
  158. void init();
  159. };
  160. inline g1_map_vertex_class *g1_vertex_min(g1_map_vertex_class *v1,g1_map_vertex_class *v2)
  161. {
  162. if (v1->height<v2->height)
  163. return v1;
  164. else return v2;
  165. }
  166. i4_bool g1_load_map_verts(g1_map_vertex_class *list, int lsize,
  167. i4_loader_class *fp, int goto_sections);
  168. void g1_save_map_verts(g1_map_vertex_class *list, int lsize,
  169. i4_saver_class *fp, int mark_section);
  170. inline g1_map_vertex_class *g1_get_vertex(int x, int y)
  171. {
  172. return g1_verts + y*(g1_map_width+1) + x;
  173. }
  174. #endif