map_lod.cc 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782
  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.hh"
  9. #include "map_man.hh"
  10. #include "map_vert.hh"
  11. #include "map_cell.hh"
  12. #include "div_table.cc"
  13. #include "camera.hh"
  14. #include "r1_api.hh"
  15. #include "tmanage.hh"
  16. #include "r1_clip.hh"
  17. #include "g1_render.hh"
  18. #include "tile.hh"
  19. #include "controller.hh"
  20. #include "statistics.hh"
  21. #include "lisp/li_class.hh"
  22. #include "resources.hh"
  23. #include "time/profile.hh"
  24. #include "light.hh"
  25. static i4_profile_class pf_map_fast("map_fast"),
  26. pf_calc_map_lod("calc_map_lod"),
  27. pf_gather_objects("gather_objects"),
  28. pf_draw_tri("draw_tri");
  29. i4_image_class *render_map_section(int x1, int y1, int x2, int y2, int im_w, int im_h);
  30. inline float fmin(float a, float b) { return (a<b)?a:b; }
  31. inline float fmax(float a, float b) { return (a>b)?a:b; }
  32. class texture_context
  33. {
  34. public:
  35. i4_float x1,y1; // offset in the map
  36. i4_float sx, sy; // scale factors to get [0,1] texture coords.
  37. r1_texture_handle texture; // texture to use
  38. };
  39. class lod_node
  40. {
  41. public:
  42. w16 x1,y1,x2,y2; // bounding box for this LOD node
  43. i4_float z1,z2;
  44. i4_float metric; // metric for determining LOD "importance"
  45. lod_node *child[2]; // orthogonal BSP children
  46. w8 texture_context; // which texture context to use or 0 for 1x1
  47. w8 flags;
  48. enum
  49. {
  50. CLIPPED = 1<<0, // this node is completely out of the view
  51. IN_VIEW = 1<<1, // in the view
  52. CLIP_FLAGS = CLIPPED | IN_VIEW,
  53. V_SPLIT = 1<<2, // node split vertically
  54. };
  55. lod_node(w16 x1,w16 y1,w16 x2,w16 y2, i4_float metric=0)
  56. : x1(x1), y1(y1), x2(x2), y2(y2), metric(metric)
  57. {
  58. child[0] = 0;
  59. child[1] = 0;
  60. }
  61. };
  62. class g1_lod_context_class
  63. {
  64. public:
  65. // texture contexts for the large subdivided texture used to cover the whole map
  66. texture_context map_texture[5];
  67. int num_context, last_context;
  68. // LOD tree
  69. lod_node *root;
  70. // breadth first ordering of the lod_nodes
  71. lod_node *queue[150*100*2];
  72. int num_queued, front;
  73. // lod_nodes submitted for drawing
  74. lod_node *quad[150*100];
  75. int num_quads;
  76. // current controller
  77. g1_object_controller_class *cont;
  78. i4_3d_vector pos;
  79. g1_lod_context_class() : root(0) {}
  80. void init_lod()
  81. // make LOD tree
  82. {
  83. if (root)
  84. delete_lod_tree(root);
  85. num_context=0;
  86. make_lod_node(&root,0,0,0,g1_get_map()->width(),g1_get_map()->height());
  87. for (int i=1; i<=num_context; i++)
  88. {
  89. int x1,y1,x2,y2;
  90. x1 = i4_f_to_i(map_texture[i].x1);
  91. y1 = i4_f_to_i(map_texture[i].y1);
  92. x2 = x1 + i4_f_to_i(1.0/map_texture[i].sx + 0.5);
  93. y2 = y1 + i4_f_to_i(1.0/map_texture[i].sy + 0.5);
  94. i4_image_class *im = render_map_section(x1,y1,x2,y2, 256,256);
  95. map_texture[i].texture = g1_render.r_api->get_tmanager()->register_image(im);
  96. delete im;
  97. }
  98. }
  99. void uninit_lod()
  100. {
  101. delete_lod_tree(root);
  102. }
  103. void use_controller(g1_object_controller_class *_cont) { cont = _cont; }
  104. int test_clip(lod_node *p)
  105. // test bounding region of lod_node against the view frustrum
  106. {
  107. const i4_float MAX_FLY_HEIGHT=3.0;
  108. int clip = cont->test_clip(i4_3d_vector(p->x1,p->y1,p->z1),
  109. i4_3d_vector(p->x2,p->y2,p->z2+MAX_FLY_HEIGHT));
  110. w8 flags =
  111. (clip<0)? lod_node::CLIPPED :
  112. (clip>0)? lod_node::IN_VIEW :
  113. 0;
  114. p->flags = (p->flags & ~lod_node::CLIP_FLAGS) | flags;
  115. return clip>=0;
  116. }
  117. void make_lod_node(lod_node **pp, lod_node *parent, int x1, int y1, int x2, int y2, int level=0)
  118. {
  119. int dx = x2-x1, dy = y2-y1;
  120. int mx = (x1+x2)/2, my = (y1+y2)/2;
  121. if (dx==0 || dy==0)
  122. return;
  123. lod_node *p;
  124. *pp = p = new lod_node(x1,y1,x2,y2);
  125. // hack to get texture contexts
  126. if (level<2)
  127. p->texture_context = 0;
  128. else if (level==2)
  129. {
  130. p->texture_context = ++num_context;
  131. map_texture[num_context].x1 = x1;
  132. map_texture[num_context].y1 = y1;
  133. map_texture[num_context].sx = 1.0*div_table[x2-x1];
  134. map_texture[num_context].sy = 1.0*div_table[y2-y1];
  135. }
  136. else
  137. p->texture_context = parent->texture_context;
  138. if (dx>1 || dy>1)
  139. {
  140. if (dx>=dy)
  141. {
  142. p->flags |= lod_node::V_SPLIT;
  143. make_lod_node(&p->child[0], p, x1,y1,mx,y2, level+1);
  144. make_lod_node(&p->child[1], p, mx,y1,x2,y2, level+1);
  145. }
  146. else
  147. {
  148. p->flags &= ~lod_node::V_SPLIT;
  149. make_lod_node(&p->child[0], p, x1,y1,x2,my, level+1);
  150. make_lod_node(&p->child[1], p, x1,my,x2,y2, level+1);
  151. }
  152. p->z1 = fmin(p->child[0]->z1, p->child[1]->z1);
  153. p->z2 = fmax(p->child[0]->z2, p->child[1]->z2);
  154. // area metric
  155. p->metric = p->child[0]->metric + p->child[1]->metric;
  156. }
  157. else
  158. {
  159. i4_float
  160. z11 = g1_get_map()->vertex(x1,y1)->get_height(),
  161. z21 = g1_get_map()->vertex(x2,y1)->get_height(),
  162. z12 = g1_get_map()->vertex(x1,y2)->get_height(),
  163. z22 = g1_get_map()->vertex(x2,y2)->get_height();
  164. p->z1 = p->z2 = z11;
  165. p->z1 = fmin(p->z1, z21);
  166. p->z2 = fmax(p->z2, z21);
  167. p->z1 = fmin(p->z1, z12);
  168. p->z2 = fmax(p->z2, z12);
  169. p->z1 = fmin(p->z1, z22);
  170. p->z2 = fmax(p->z2, z22);
  171. // area metric
  172. i4_float dz1,dz2;
  173. dz1 = z22 - z21;
  174. dz2 = z11 - z21;
  175. p->metric = sqrt(dz1*dz1 + dz2*dz2 + 1);
  176. dz1 = z22 - z12;
  177. dz2 = z11 - z12;
  178. p->metric += dz1*dz1 + dz2*dz2 + 1;
  179. }
  180. }
  181. void delete_lod_tree(lod_node *p)
  182. {
  183. if (!p)
  184. return;
  185. delete_lod_tree(p->child[0]);
  186. delete_lod_tree(p->child[1]);
  187. delete p;
  188. }
  189. void calculate_lod(lod_node *p)
  190. {
  191. // cont->view.get_camera_pos(pos);
  192. cont->get_pos(pos);
  193. num_quads=0;
  194. front = num_queued = 0;
  195. queue[num_queued++] = p;
  196. root->flags = 0;
  197. while (front<num_queued)
  198. {
  199. p = queue[front++];
  200. int x1=p->x1,y1=p->y1,x2=p->x2,y2=p->y2;
  201. w8 sub_flags=p->flags & lod_node::IN_VIEW;
  202. // do some kind of clip test
  203. if (!sub_flags)
  204. if (test_clip(p))
  205. sub_flags=p->flags & lod_node::IN_VIEW;
  206. else
  207. continue;
  208. // int dx = x2-x1, dy = y2-y1;
  209. int mx = (x1+x2)/2, my = (y1+y2)/2;
  210. i4_float fx = pos.x - mx, fy = pos.y - my, fz = pos.z - g1_get_vertex(mx,my)->get_height();
  211. i4_float d1 = fx*fx+fy*fy+fz*fz;
  212. i4_float d2 = p->metric*20;
  213. if (p->child[0] && d1<d2)
  214. {
  215. p->child[0]->flags = (p->child[0]->flags & ~lod_node::CLIP_FLAGS) | sub_flags;
  216. p->child[1]->flags = (p->child[1]->flags & ~lod_node::CLIP_FLAGS) | sub_flags;
  217. queue[num_queued++] = p->child[0];
  218. queue[num_queued++] = p->child[1];
  219. // count & mark t splits
  220. lod_node *q = p->child[0];
  221. if (p->flags & lod_node::V_SPLIT)
  222. {
  223. g1_get_vertex(q->x2,q->y1)->flags ^= g1_map_vertex_class::T_INTERSECTION;
  224. g1_get_vertex(q->x2,q->y2)->flags ^= g1_map_vertex_class::T_INTERSECTION;
  225. }
  226. else
  227. {
  228. g1_get_vertex(q->x1,q->y2)->flags ^= g1_map_vertex_class::T_INTERSECTION;
  229. g1_get_vertex(q->x2,q->y2)->flags ^= g1_map_vertex_class::T_INTERSECTION;
  230. }
  231. }
  232. else
  233. {
  234. g1_map_vertex_class *v11, *v12, *v21, *v22;
  235. v11 = g1_get_vertex(x1,y1);
  236. v21 = g1_get_vertex(x2,y1);
  237. v12 = g1_get_vertex(x1,y2);
  238. v22 = g1_get_vertex(x2,y2);
  239. v11->t_height = v11->get_height();
  240. v21->t_height = v21->get_height();
  241. v12->t_height = v12->get_height();
  242. v22->t_height = v22->get_height();
  243. quad[num_quads++] = p;
  244. }
  245. }
  246. // T intersection fixing
  247. // Map of directions
  248. // dzdx2
  249. // z22* ----> *
  250. // ^ ^
  251. // dzdy1 | | dzdy2
  252. // | |
  253. // z12* ----> *
  254. // z11 dzdx1 z21
  255. for (int test=0; test<2; test++)
  256. for (front = 0; front<num_quads; front++)
  257. {
  258. int
  259. x1=quad[front]->x1,y1=quad[front]->y1,
  260. x2=quad[front]->x2,y2=quad[front]->y2;
  261. int i,j;
  262. i4_float z11,z12,z21,z22;
  263. i4_float dzdx1,dzdx2,dzdy1,dzdy2;
  264. z11 = g1_get_vertex(x1,y1)->t_height;
  265. z21 = g1_get_vertex(x2,y1)->t_height;
  266. z12 = g1_get_vertex(x1,y2)->t_height;
  267. z22 = g1_get_vertex(x2,y2)->t_height;
  268. dzdx1 = (z21 - z11)*div_table[x2 - x1];
  269. dzdx2 = (z22 - z12)*div_table[x2 - x1];
  270. dzdy1 = (z12 - z11)*div_table[y2 - y1];
  271. dzdy2 = (z22 - z21)*div_table[y2 - y1];
  272. z22 = z12;
  273. z12 = z11;
  274. i4_float adj1,adj2;
  275. #define DEPIXEL_ADJ 0.1
  276. adj1 = (pos.x<x1)?DEPIXEL_ADJ:-DEPIXEL_ADJ;
  277. adj2 = (pos.x>x2)?DEPIXEL_ADJ:-DEPIXEL_ADJ;
  278. for (j=y1+1; j<y2; j++)
  279. {
  280. z11 += dzdy1;
  281. z21 += dzdy2;
  282. // if (g1_get_vertex(x1,j)->flags & g1_map_vertex_class::T_INTERSECTION)
  283. g1_get_vertex(x1,j)->t_height = z11 + adj1;
  284. // if (g1_get_vertex(x2,j)->flags & g1_map_vertex_class::T_INTERSECTION)
  285. g1_get_vertex(x2,j)->t_height = z21 + adj2;
  286. }
  287. adj1 = (pos.y<y1)?DEPIXEL_ADJ:-DEPIXEL_ADJ;
  288. adj2 = (pos.y>y2)?DEPIXEL_ADJ:-DEPIXEL_ADJ;
  289. for (i=x1+1; i<x2; i++)
  290. {
  291. z12 += dzdx1;
  292. z22 += dzdx2;
  293. // if (g1_get_vertex(i,y1)->flags & g1_map_vertex_class::T_INTERSECTION)
  294. g1_get_vertex(i,y1)->t_height = z12 + adj1;
  295. // if (g1_get_vertex(i,y2)->flags & g1_map_vertex_class::T_INTERSECTION)
  296. g1_get_vertex(i,y2)->t_height = z22 + adj2;
  297. }
  298. }
  299. }
  300. } g1_lod;
  301. void g1_map_class::init_lod()
  302. {
  303. g1_lod.init_lod();
  304. }
  305. void g1_map_class::calc_map_lod(g1_object_controller_class *cont)
  306. {
  307. pf_calc_map_lod.start();
  308. g1_lod.use_controller(cont);
  309. g1_lod.calculate_lod(g1_lod.root);
  310. pf_calc_map_lod.stop();
  311. }
  312. static r1_texture_ref g1_default_texture("tron_grid");
  313. const w32 g1_max_objs_in_view = 256;
  314. w32 g1_num_objs_in_view = 0;
  315. w32 g1_objs_in_view[g1_max_objs_in_view];
  316. r1_vert temp_buf_1[9];
  317. r1_vert temp_buf_2[9];
  318. i4_array<i4_transform_class> g1_obj_transforms_in_view(0, g1_max_objs_in_view);
  319. class transform_killer_class : public i4_init_class
  320. {
  321. //fix this trey! can't have global i4_array's without something to clean it up.
  322. void uninit() { g1_obj_transforms_in_view.uninit(); }
  323. } transform_killer;
  324. i4_bool g1_draw_tri(r1_vert *points,
  325. w16 a, w16 b, w16 c,
  326. r1_texture_handle texture, w16 clip=1)
  327. //{{{
  328. {
  329. pf_draw_tri.start();
  330. g1_stat_counter.increment(g1_statistics_counter_class::TERRAIN_POLYS);
  331. g1_stat_counter.increment(g1_statistics_counter_class::TOTAL_POLYS);
  332. i4_bool ret = i4_F;
  333. r1_render_api_class *api = g1_render.r_api;
  334. i4_float size=0;
  335. sw32 num_poly_verts = 3;
  336. r1_vert *clipped_poly, *v;
  337. w16 *clipped_refs;
  338. static w16 clipping_refs[16] = { 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15 };
  339. w16 refs[3];
  340. i4_float near_w=0.0001;
  341. int i,j;
  342. refs[0] = a;
  343. refs[1] = b;
  344. refs[2] = c;
  345. clipped_poly = points;
  346. clipped_refs = &refs[0];
  347. // hacked backface culling (should do this in worldspace before projection...)
  348. i4_3d_vector vt1(clipped_poly[a].v),vt2(clipped_poly[c].v),normal;
  349. vt1 -= clipped_poly[b].v;
  350. vt2 -= clipped_poly[b].v;
  351. normal.cross(vt2,vt1);
  352. if (normal.dot(clipped_poly[b].v)>0)
  353. {
  354. pf_draw_tri.stop();
  355. return i4_F;
  356. }
  357. if ((clipped_poly[a].outcode | clipped_poly[b].outcode | clipped_poly[c].outcode)==0)
  358. {
  359. for (j=0; j<num_poly_verts; j++)
  360. {
  361. v = &clipped_poly[clipped_refs[j]];
  362. if (v->w > near_w)
  363. near_w=v->w;
  364. }
  365. }
  366. else
  367. {
  368. clipped_poly = api->clip_poly(&num_poly_verts,
  369. clipped_poly,
  370. clipped_refs,
  371. temp_buf_1,
  372. temp_buf_2,
  373. R1_CLIP_NO_CALC_OUTCODE);
  374. if (!clipped_poly || num_poly_verts<3)
  375. num_poly_verts=0;
  376. if (num_poly_verts)
  377. {
  378. clipped_refs = clipping_refs;
  379. for (j=0; j<num_poly_verts; j++)
  380. {
  381. v = &clipped_poly[j];
  382. float ooz = r1_ooz(v->v.z);
  383. v->px = v->v.x*ooz*g1_render.center_x + g1_render.center_x;
  384. v->py = v->v.y*ooz*g1_render.center_y + g1_render.center_y;
  385. v->w = ooz;
  386. if (v->w > near_w)
  387. near_w=v->w;
  388. }
  389. }
  390. }
  391. if (num_poly_verts)
  392. {
  393. if (texture)
  394. {
  395. sw32 texture_size=i4_f_to_i(g1_render.center_x * near_w * 0.5 * 0.5);
  396. api->use_texture(texture, texture_size, 0);
  397. }
  398. if (g1_render.draw_mode==g1_render_class::WIREFRAME)
  399. {
  400. api->set_constant_color(0x7f7f7f);
  401. api->disable_texture();
  402. api->set_shading_mode(R1_SHADE_DISABLED);
  403. r1_vert v[4];
  404. v[0]=clipped_poly[0];
  405. v[1]=clipped_poly[1];
  406. v[2]=clipped_poly[2];
  407. v[3]=clipped_poly[0];
  408. r1_clip_render_lines(3, v, g1_render.center_x, g1_render.center_y, api);
  409. api->set_shading_mode(R1_WHITE_SHADING);
  410. }
  411. else
  412. api->render_poly(num_poly_verts, clipped_poly, clipped_refs);
  413. ret = i4_T;
  414. }
  415. pf_draw_tri.stop();
  416. return ret;
  417. }
  418. static void gather_objects(int x1, int y1, int x2, int y2)
  419. {
  420. pf_gather_objects.start();
  421. for (int y=y1; y<y2; y++)
  422. {
  423. g1_map_cell_class *map_cell=g1_get_cell(x1,y);
  424. for (int x=x1; x<x2; x++, map_cell++)
  425. {
  426. // collect objects
  427. for (g1_object_chain_class *o=map_cell->get_obj_list(); o; o=o->next)
  428. {
  429. g1_object_class *obj=o->object;
  430. if (!obj->get_flag(g1_object_class::SCRATCH_BIT))
  431. {
  432. if (g1_num_objs_in_view<g1_max_objs_in_view)
  433. {
  434. g1_objs_in_view[g1_num_objs_in_view] = obj->global_id;
  435. obj->world_transform = g1_obj_transforms_in_view.add();
  436. g1_num_objs_in_view++;
  437. //have the object update his transform
  438. obj->calc_world_transform(g1_render.frame_ratio);
  439. obj->set_flag(g1_object_class::SCRATCH_BIT, 1);
  440. }
  441. }
  442. }
  443. }
  444. }
  445. pf_gather_objects.stop();
  446. }
  447. static i4_transform_class comp_t;
  448. int object_compare(const void *a, const void *b)
  449. {
  450. g1_object_class
  451. *oa = g1_global_id.get(*((w32*)a)),
  452. *ob = g1_global_id.get(*((w32*)b));
  453. // sort objects with alpha last
  454. if (oa->get_type()->get_flag(g1_object_definition_class::HAS_ALPHA))
  455. {
  456. if (!ob->get_type()->get_flag(g1_object_definition_class::HAS_ALPHA))
  457. return -1;
  458. }
  459. else if (ob->get_type()->get_flag(g1_object_definition_class::HAS_ALPHA))
  460. return 1;
  461. i4_float
  462. za = comp_t.x.z*oa->x + comp_t.y.z*oa->y + comp_t.z.z*oa->h + comp_t.t.z,
  463. zb = comp_t.x.z*ob->x + comp_t.y.z*ob->y + comp_t.z.z*ob->h + comp_t.t.z;
  464. if (za>zb)
  465. return 1;
  466. else
  467. return -1;
  468. }
  469. void g1_map_class::fast_draw_cells(g1_draw_context_class *context)
  470. {
  471. pf_map_fast.start();
  472. r1_render_api_class *api = g1_render.r_api;
  473. i4_transform_class t(*context->transform);
  474. i4_3d_vector pos;
  475. // g1_lod.cont->view.get_camera_pos(pos);
  476. pos = g1_lod.pos;
  477. g1_num_objs_in_view = 0;
  478. g1_obj_transforms_in_view.clear();
  479. if (g1_render.draw_mode==g1_render_class::SOLID)
  480. api->use_texture(g1_default_texture.get(), 1, 0);
  481. int i=0,j;
  482. g1_map_vertex_class *vt[4];
  483. g1_map_vertex_class *v=verts;
  484. g1_lod.last_context=0;
  485. for (j=0; j<g1_lod.num_quads; j++)
  486. {
  487. lod_node *p = g1_lod.quad[j];
  488. int
  489. x1 = p->x1, y1 = p->y1,
  490. x2 = p->x2, y2 = p->y2,
  491. clip = !(p->flags & lod_node::IN_VIEW);
  492. sw32 mw_p1=width()+1;
  493. vt[0]=v + x1 + y1 * mw_p1;
  494. vt[1]=v + x2 + y1 * mw_p1;
  495. vt[2]=v + x2 + y2 * mw_p1;
  496. vt[3]=v + x1 + y2 * mw_p1;
  497. vt[0]->transform(t, x1, y1, g1_render.scale_x,g1_render.scale_y);
  498. vt[1]->transform(t, x2, y1, g1_render.scale_x,g1_render.scale_y);
  499. vt[2]->transform(t, x2, y2, g1_render.scale_x,g1_render.scale_y);
  500. vt[3]->transform(t, x1, y2, g1_render.scale_x,g1_render.scale_y);
  501. if (vt[0]->calc_clip_code()==0)
  502. vt[0]->project(g1_render.center_x, g1_render.center_y);
  503. if (vt[1]->calc_clip_code()==0)
  504. vt[1]->project(g1_render.center_x, g1_render.center_y);
  505. if (vt[2]->calc_clip_code()==0)
  506. vt[2]->project(g1_render.center_x, g1_render.center_y);
  507. if (vt[3]->calc_clip_code()==0)
  508. vt[3]->project(g1_render.center_x, g1_render.center_y);
  509. // collect objects
  510. i4_3d_vector mid(i4_float(x2+x1)/2.0,i4_float(y2+y1)/2.0,i4_float(p->z2+p->z1)/2.0);
  511. mid -= pos;
  512. if (mid.dot(mid) < g1_resources.lod_disappear_dist)
  513. gather_objects(x1,y1,x2,y2);
  514. else
  515. i++;
  516. if (x1+1==x2 && y1+1==y2)
  517. {
  518. // single cell
  519. g1_map_cell_class *map_cell=cells + x1 + y1*width();
  520. g1_lod.last_context=0;
  521. r1_texture_handle han = g1_tile_man.get_texture(map_cell->type);
  522. if (han && han!=g1_tile_man.get_pink())
  523. {
  524. static float u[4]={0,1,1,0},v[4]={1,1,0,0};
  525. int uv_on=map_cell->get_rotation();
  526. int uv_dir=map_cell->mirrored() ? 3 : 1;
  527. r1_vert poly[4];
  528. for (int i=0; i<4; i++)
  529. {
  530. vt[i]->set_r1_vert(&poly[i]);
  531. poly[i].s=u[uv_on];
  532. poly[i].t=v[uv_on];
  533. poly[i].a=1.0;
  534. uv_on=(uv_on+uv_dir)&3;
  535. }
  536. vt[0]->get_rgb(poly[0].r, poly[0].g, poly[0].b, x1, y1);
  537. vt[1]->get_rgb(poly[1].r, poly[1].g, poly[1].b, x2, y1);
  538. vt[2]->get_rgb(poly[2].r, poly[2].g, poly[2].b, x2, y2);
  539. vt[3]->get_rgb(poly[3].r, poly[3].g, poly[3].b, x1, y2);
  540. if (post_cell_draw)
  541. post_cell_draw(x1,y1, post_cell_draw_context);
  542. if (g1_draw_tri(poly, 0,1,2, han))
  543. han = 0;
  544. g1_draw_tri(poly, 0,2,3, han);
  545. }
  546. }
  547. else
  548. {
  549. // larger LOD cell
  550. texture_context *tc = &g1_lod.map_texture[p->texture_context];
  551. r1_texture_handle han;
  552. if (g1_lod.last_context==p->texture_context)
  553. han = 0;
  554. else
  555. han = tc->texture;
  556. r1_vert poly[4];
  557. for (int i=0; i<4; i++)
  558. {
  559. vt[i]->set_r1_vert(&poly[i]);
  560. poly[i].a=1.0;
  561. }
  562. vt[0]->get_rgb(poly[0].r, poly[0].g, poly[0].b, x1, y1);
  563. vt[1]->get_rgb(poly[1].r, poly[1].g, poly[1].b, x2, y1);
  564. vt[2]->get_rgb(poly[2].r, poly[2].g, poly[2].b, x2, y2);
  565. vt[3]->get_rgb(poly[3].r, poly[3].g, poly[3].b, x1, y2);
  566. poly[0].s=(x1 - tc->x1)*tc->sx;
  567. poly[0].t=(y1 - tc->y1)*tc->sy;
  568. poly[1].s=(x2 - tc->x1)*tc->sx;
  569. poly[1].t=(y1 - tc->y1)*tc->sy;
  570. poly[2].s=(x2 - tc->x1)*tc->sx;
  571. poly[2].t=(y2 - tc->y1)*tc->sy;
  572. poly[3].s=(x1 - tc->x1)*tc->sx;
  573. poly[3].t=(y2 - tc->y1)*tc->sy;
  574. if (g1_draw_tri(poly, 0,1,2, han))
  575. han = 0;
  576. if (g1_draw_tri(poly, 0,2,3, han))
  577. han = 0;
  578. if (han==0)
  579. g1_lod.last_context = p->texture_context;
  580. }
  581. }
  582. for (i=0; i<g1_num_objs_in_view; i++)
  583. {
  584. g1_object_class *o=g1_global_id.checked_get(g1_objs_in_view[i]);
  585. if (o)
  586. o->set_flag(g1_object_class::SCRATCH_BIT, 0);
  587. }
  588. li_class *old_this=li_this;
  589. //draw the objects BACK TO FRONT
  590. comp_t = *context->transform;
  591. qsort(g1_objs_in_view, g1_num_objs_in_view, sizeof(g1_objs_in_view[0]), object_compare);
  592. for (i=g1_num_objs_in_view-1;i>=0;i--)
  593. {
  594. g1_object_class *o=g1_global_id.checked_get(g1_objs_in_view[i]);
  595. if (o)
  596. {
  597. li_this=o->vars;
  598. o->set_flag(g1_object_class::SCRATCH_BIT, 0);
  599. if (o->world_transform!=0)
  600. o->draw(context);
  601. else
  602. i4_warning("null transform");
  603. }
  604. }
  605. if (context->draw_editor_stuff)
  606. {
  607. for (i=g1_num_objs_in_view-1;i>=0;i--)
  608. {
  609. g1_object_class *o=g1_global_id.checked_get(g1_objs_in_view[i]);
  610. if (o)
  611. {
  612. li_this=o->vars;
  613. o->editor_draw(context);
  614. }
  615. }
  616. }
  617. li_this=old_this;
  618. for (j=0; j<g1_lod.num_quads; j++)
  619. {
  620. lod_node *p = g1_lod.quad[j];
  621. int
  622. x1 = p->x1, y1 = p->y1,
  623. x2 = p->x2, y2 = p->y2;
  624. sw32 mw_p1=width()+1;
  625. vt[0]=v + x1 + y1 * mw_p1;
  626. vt[1]=v + x2 + y1 * mw_p1;
  627. vt[2]=v + x2 + y2 * mw_p1;
  628. vt[3]=v + x1 + y2 * mw_p1;
  629. vt[0]->clear_calculations();
  630. vt[1]->clear_calculations();
  631. vt[2]->clear_calculations();
  632. vt[3]->clear_calculations();
  633. }
  634. g1_stat_counter.increment(g1_statistics_counter_class::FRAMES);
  635. pf_map_fast.stop();
  636. }
  637. class num_in_view_reset_class : public g1_global_id_reset_notifier
  638. {
  639. public:
  640. virtual void reset()
  641. {
  642. g1_num_objs_in_view=0;
  643. }
  644. } num_in_view_reset_inst;