map_vis.cc 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  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 "map_cell.hh"
  9. #include <stdlib.h>
  10. #include "poly/polyclip.hh"
  11. #include "poly/polydraw.hh"
  12. #include "map.hh"
  13. #include "math/transform.hh"
  14. #include "time/profile.hh"
  15. #include "r1_api.hh"
  16. #include "g1_render.hh"
  17. #include "tile.hh"
  18. #include "map_vert.hh"
  19. #include "map_view.hh"
  20. static i4_profile_class pf_map_calc_visible("map::calc_visible");
  21. static int cell_compare(const void *a, const void *b)
  22. {
  23. g1_map_vertex_class *v1= g1_verts + ((g1_visible_cell *)a)->x +
  24. ((g1_visible_cell *)a)->y * g1_map_width_plus_one;
  25. g1_map_vertex_class *v2= g1_verts + ((g1_visible_cell *)b)->x +
  26. ((g1_visible_cell *)b)->y * g1_map_width_plus_one;
  27. if (v1->v.z<v2->v.z)
  28. return 1;
  29. else if (v1->v.z>v2->v.z)
  30. return -1;
  31. else return 0;
  32. }
  33. class g1_map_clip_class
  34. {
  35. public:
  36. i4_float cx2, cy2;
  37. g1_map_clip_class(i4_float cx2, i4_float cy2)
  38. : cx2(cx2), cy2(cy2) {}
  39. void project(i4_vertex_class *v)
  40. {
  41. v->px=v->v.x;
  42. v->py=v->v.y;
  43. }
  44. // used by poly_clip template function
  45. i4_float intersect(i4_vertex_class *v1, i4_vertex_class *v2, int plane)
  46. {
  47. switch (plane)
  48. {
  49. case I4_CLIP_PLANE_Z1 :
  50. case I4_CLIP_PLANE_Z2 :
  51. return 0;
  52. case I4_CLIP_PLANE_Y1 :
  53. {
  54. if (v1->v.y < v2->v.y)
  55. return (0.0-v1->v.y)/(v2->v.y-v1->v.y);
  56. else
  57. return (v1->v.y)/(v1->v.y-v2->v.y);
  58. } break;
  59. case I4_CLIP_PLANE_Y2 :
  60. {
  61. if (v1->v.y < v2->v.y)
  62. return (cy2-v1->v.y)/(v2->v.y-v1->v.y);
  63. else
  64. return (v1->v.y-cy2)/(v1->v.y-v2->v.y);
  65. } break;
  66. case I4_CLIP_PLANE_X1 :
  67. {
  68. if (v1->v.x < v2->v.x)
  69. return (0.0-v1->v.x)/(v2->v.x-v1->v.x);
  70. else
  71. return (v1->v.x)/(v1->v.x-v2->v.x);
  72. } break;
  73. case I4_CLIP_PLANE_X2 :
  74. {
  75. if (v1->v.x < v2->v.x)
  76. return (cx2-v1->v.x)/(v2->v.x-v1->v.x);
  77. else
  78. return (v1->v.x-cx2)/(v1->v.x-v2->v.x);
  79. } break;
  80. }
  81. return 0;
  82. }
  83. };
  84. class g1_vert_transformer
  85. {
  86. public:
  87. w32 max_cells;
  88. w32 t_cells;
  89. g1_visible_cell *cell_list;
  90. i4_transform_class t;
  91. g1_map_vertex_class *verts;
  92. int map_width_p1;
  93. i4_float xscale, yscale;
  94. void add_cell(w8 x, w8 y)
  95. {
  96. if (t_cells<max_cells)
  97. {
  98. cell_list[t_cells].x=x;
  99. cell_list[t_cells].y=y;
  100. t_cells++;
  101. }
  102. }
  103. g1_vert_transformer(i4_transform_class &trans,
  104. g1_map_vertex_class *verts,
  105. int map_width_p1,
  106. g1_visible_cell *cell_list,
  107. w32 max_cells,
  108. i4_float xscale, i4_float yscale)
  109. : verts(verts), map_width_p1(map_width_p1),
  110. cell_list(cell_list), max_cells(max_cells),
  111. xscale(xscale), yscale(yscale)
  112. {
  113. t=trans;
  114. t_cells=0;
  115. }
  116. int feature(int x) { return 0; }
  117. class screen_pointer
  118. {
  119. public:
  120. i4_3d_vector ground;
  121. g1_map_vertex_class *v;
  122. w8 x,y;
  123. screen_pointer& operator=(const screen_pointer& p)
  124. {
  125. ground=p.ground;
  126. v=p.v;
  127. x=p.x;
  128. y=p.y;
  129. return *this;
  130. }
  131. void move_to(int _x, int _y, g1_vert_transformer &vp)
  132. {
  133. x=_x;
  134. y=_y;
  135. i4_3d_point_class g=i4_3d_point_class(_x,_y,0);
  136. vp.t.transform(g, ground);
  137. v=vp.verts + _x + _y * vp.map_width_p1;
  138. }
  139. void add(int amount, g1_vert_transformer &vp)
  140. {
  141. x+=amount;
  142. ground.x += vp.t.x.x * amount;
  143. ground.y += vp.t.x.y * amount;
  144. ground.z += vp.t.x.z * amount;
  145. v+=amount;
  146. }
  147. void next_line(g1_vert_transformer &vp)
  148. {
  149. y++;
  150. ground.x+=vp.t.y.x;
  151. ground.y+=vp.t.y.y;
  152. ground.z+=vp.t.y.z;
  153. v+=vp.map_width_p1;
  154. }
  155. int get_twidth(g1_vert_transformer &ref)
  156. {
  157. return 0;
  158. }
  159. w16 * get_ptr(g1_vert_transformer &ref)
  160. {
  161. return NULL;
  162. }
  163. i4_float * get_lookup(g1_vert_transformer &ref)
  164. {
  165. return NULL;
  166. }
  167. w16 * get_tptr(g1_vert_transformer &ref)
  168. {
  169. return NULL;
  170. }
  171. w32 get_screen_width(g1_vert_transformer &ref)
  172. {
  173. return 0;
  174. }
  175. void write(int s, int t,
  176. i4_float w,
  177. i4_float r, i4_float g, i4_float b, i4_float a,
  178. g1_vert_transformer &vp)
  179. {
  180. if ((v->flags & g1_map_vertex_class::APPLY_WAVE_FUNCTION)==0)
  181. {
  182. i4_float h=v->get_height();
  183. i4_3d_vector v_air;
  184. v_air.x = (ground.x + h * vp.t.z.x) * vp.xscale;
  185. v_air.y = (ground.y + h * vp.t.z.y) * vp.yscale;
  186. v_air.z = ground.z + h * vp.t.z.z;
  187. // v->transform(vp.t, x,y);
  188. v->v=v_air;
  189. }
  190. else
  191. {
  192. v->wave_transform(vp.t, x, y);
  193. v->v.x *= vp.xscale;
  194. v->v.y *= vp.yscale;
  195. }
  196. v->set_is_transformed(1);
  197. vp.add_cell(x,y);
  198. }
  199. };
  200. };
  201. int g1_map_class::calc_visible(i4_transform_class &t,
  202. i4_polygon_class *area_poly,
  203. g1_visible_cell *buffer, w32 buf_elements,
  204. i4_float xscale, i4_float yscale)
  205. {
  206. pf_map_calc_visible.start();
  207. int ret=0,i;
  208. i4_polygon_class poly, clipped_poly;
  209. for (i =0; i<area_poly->t_verts; i++)
  210. {
  211. area_poly->vert[i].px=area_poly->vert[i].v.x;
  212. area_poly->vert[i].py=area_poly->vert[i].v.y;
  213. }
  214. poly=*area_poly;
  215. g1_map_clip_class cell_clip(width(), height());
  216. i4_poly_clip(cell_clip, poly, clipped_poly,
  217. (float)0.0,(float)0.0,(float) -1.0,
  218. (float)(width()),
  219. (float)(height()), (float)1.0);
  220. if (clipped_poly.t_verts>=3)
  221. {
  222. g1_vert_transformer vt(t,
  223. verts, width()+1,
  224. buffer, buf_elements,
  225. xscale, yscale);
  226. i4_poly_draw_class<g1_vert_transformer> draw(vt, clipped_poly);
  227. ret=vt.t_cells;
  228. }
  229. for (i=0; i<ret; i++)
  230. if (buffer[i].x>=width() || buffer[i].y>=height())
  231. {
  232. i4_warning("bad cell");
  233. return 0;
  234. }
  235. g1_radar_looking_at(clipped_poly.vert[0].v.x,
  236. clipped_poly.vert[0].v.y,
  237. clipped_poly.vert[2].v.x,
  238. clipped_poly.vert[2].v.y);
  239. if (ret)
  240. {
  241. //if we're using a software rasterizer, must sort the cells (technically not necessary anymore)
  242. //if (g1_render.r_api->get_render_device_flags() & R1_SOFTWARE)
  243. //qsort(buffer, ret, sizeof(g1_visible_cell), cell_compare);
  244. //else
  245. //sort the cells by texture
  246. //qsort(buffer, ret, sizeof(g1_visible_cell), cell_texture_compare);
  247. }
  248. pf_map_calc_visible.stop();
  249. return ret;
  250. }