map_cell.cc 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  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_man.hh"
  9. #include "map_cell.hh"
  10. #include "g1_object.hh"
  11. #include "tile.hh"
  12. #include "map.hh"
  13. #include "compress/rle.hh"
  14. #include "loaders/dir_save.hh"
  15. #include "loaders/dir_load.hh"
  16. #include "saver_id.hh"
  17. #include "map_vert.hh"
  18. g1_object_chain_class *g1_map_cell_class::get_solid_list()
  19. {
  20. if (!object_list || !object_list->object->get_flag(g1_object_class::BLOCKING))
  21. return 0;
  22. else return object_list;
  23. }
  24. g1_object_chain_class *g1_map_cell_class::get_non_solid_list()
  25. {
  26. g1_object_chain_class *p=object_list;
  27. while (p && p->object->get_flag(g1_object_class::BLOCKING))
  28. p=p->next;
  29. return p;
  30. }
  31. void g1_map_cell_class::add_object(g1_object_chain_class &o)
  32. {
  33. if (!object_list ||
  34. o.object->get_flag(g1_object_class::BLOCKING)) // add solids to front of list
  35. {
  36. o.next=object_list;
  37. object_list=&o;
  38. }
  39. else
  40. {
  41. // add non solids after all solids
  42. g1_object_chain_class *last=object_list, *p=object_list->next;
  43. while (p && p->object->get_flag(g1_object_class::BLOCKING))
  44. {
  45. last=p;
  46. p=p->next;
  47. }
  48. o.next=last->next;
  49. last->next=&o;
  50. }
  51. }
  52. i4_bool g1_map_cell_class::remove_object(g1_object_chain_class &o)
  53. {
  54. if (&o==object_list)
  55. {
  56. object_list=o.next;
  57. return i4_T;
  58. }
  59. else
  60. {
  61. g1_object_chain_class *c=object_list, *last;
  62. while (c && c!=&o)
  63. {
  64. last=c;
  65. c=c->next;
  66. }
  67. if (!c)
  68. {
  69. i4_warning("map_cell::remove_object, object not in cell");
  70. return i4_F;
  71. }
  72. last->next=o.next;
  73. return i4_T;
  74. }
  75. }
  76. void g1_map_cell_class::init(w16 _type, g1_rotation_type rot,
  77. i4_bool mirrored)
  78. {
  79. object_list=0;
  80. type=_type;
  81. flags=FOGGED;
  82. top_left_normal=0x8000;
  83. bottom_right_normal=0x8000;
  84. g1_tile_man.get(_type)->apply_to_cell(*this);
  85. set_rotation(rot);
  86. if (mirrored)
  87. flags|=MIRRORED;
  88. }
  89. void g1_map_cell_class::recalc_top_left(int cx, int cy)
  90. {
  91. int mw=g1_get_map()->width(), mh=g1_get_map()->height();
  92. I4_ASSERT(cx<=mw && cy<=mh, "recalc normal : vert out of range");
  93. if (cx==mw-1 || cy==mh-1)
  94. top_left_normal=(31) | (15<<5) | (15<<10); // normal = 0,0,1
  95. else
  96. {
  97. g1_map_vertex_class *v=g1_get_map()->vertex(cx,cy);
  98. float h0=v->get_height(), h2=v[mw+1].get_height(), h3=v[mw+1+1].get_height();
  99. i4_3d_vector v1(1, 0, h3-h2), v2(0, -1, h0-h2);
  100. i4_3d_vector normal;
  101. normal.cross(v2,v1);
  102. normal.normalize();
  103. top_left_normal=g1_normal_to_16(normal);
  104. }
  105. }
  106. void g1_map_cell_class::recalc_bottom_right(int cx, int cy)
  107. {
  108. int mw=g1_get_map()->width(), mh=g1_get_map()->height();
  109. if (cx>=mw || cy>=mh)
  110. bottom_right_normal=(31) | (15<<5) | (15<<10); // normal = 0,0,1
  111. else
  112. {
  113. g1_map_vertex_class *v=g1_get_map()->vertex(cx,cy);
  114. float h0=v->get_height(), h1=v[1].get_height(), h3=v[mw+1+1].get_height();
  115. i4_3d_vector v1(-1, 0, h0-h1), v2(0, 1, h3-h1);
  116. i4_3d_vector normal;
  117. normal.cross(v2,v1);
  118. normal.normalize();
  119. bottom_right_normal=g1_normal_to_16(normal);
  120. }
  121. }
  122. void g1_save_map_cells(g1_map_cell_class *list,
  123. int lsize, i4_saver_class *fp)
  124. {
  125. int i,j;
  126. i4_rle_class<w16> fp16(fp);
  127. fp->mark_section("golgotha cell type v1");
  128. for (i=0; i<lsize; i++)
  129. fp16.write(list[i].type);
  130. fp16.flush();
  131. fp->mark_section("golgotha cell flags v2");
  132. for (i=0; i<lsize; i++)
  133. fp16.write(list[i].flags & g1_map_cell_class::SAVED_FLAGS);
  134. fp16.flush();
  135. fp->mark_section("golgotha cell normals v1");
  136. for (i=0; i<lsize; i++)
  137. fp16.write(list[i].top_left_normal);
  138. for (i=0; i<lsize; i++)
  139. fp16.write(list[i].bottom_right_normal);
  140. fp16.flush();
  141. }
  142. i4_bool g1_load_map_cells(g1_map_cell_class *list, int lsize,
  143. w16 *tile_remap, i4_loader_class *fp)
  144. {
  145. int i;
  146. i4_rle_class<w16> fp16(fp);
  147. //////////////////////////////////////////////////////////
  148. // support for old formats
  149. if (!fp->goto_section("golgotha cell type v1"))
  150. {
  151. g1_get_map()->mark_for_recalc(G1_RECALC_STATIC_LIGHT);
  152. if (fp->goto_section(OLD_G1_SECTION_CELL_V6))
  153. {
  154. for (i=0; i<lsize; i++)
  155. list[i].load_v6(fp, tile_remap);
  156. } else if (fp->goto_section(OLD_G1_SECTION_CELL_V5))
  157. {
  158. for (i=0; i<lsize; i++)
  159. list[i].load_v5(fp, tile_remap);
  160. }
  161. else if (fp->goto_section(OLD_G1_SECTION_CELL_V4))
  162. {
  163. for (i=0; i<lsize; i++)
  164. list[i].load_v4(fp, tile_remap);
  165. } else if (fp->goto_section(OLD_G1_SECTION_CELL_V3))
  166. {
  167. for (i=0; i<lsize; i++)
  168. list[i].load_v3(fp, tile_remap);
  169. } else if (fp->goto_section(OLD_G1_SECTION_CELL_V2))
  170. {
  171. for (i=0; i<lsize; i++)
  172. list[i].load_v2(fp, tile_remap);
  173. }
  174. else
  175. return i4_F;
  176. }
  177. else /////////////////////// new format : stored in rle arrays rather than structs
  178. {
  179. for (i=0; i<lsize; i++)
  180. list[i].type=tile_remap[fp16.read()];
  181. }
  182. if (fp->goto_section("golgotha cell flags v2"))
  183. {
  184. for (i=0; i<lsize; i++)
  185. list[i].flags = fp16.read() & g1_map_cell_class::SAVED_FLAGS;
  186. }
  187. else if (fp->goto_section("golgotha cell flags v1"))
  188. {
  189. for (i=0; i<lsize; i++)
  190. {
  191. list[i].flags = (fp16.read() & g1_map_cell_class::SAVED_FLAGS) | g1_map_cell_class::FOGGED;
  192. list[i].clear_restore_bits();
  193. }
  194. }
  195. for (i=0; i<lsize; i++)
  196. g1_tile_man.get(list[i].type)->apply_to_cell(list[i]);
  197. if (fp->goto_section("golgotha cell normals v1"))
  198. {
  199. for (i=0; i<lsize; i++)
  200. list[i].top_left_normal=fp16.read();
  201. for (i=0; i<lsize; i++)
  202. list[i].bottom_right_normal=fp16.read();
  203. }
  204. return i4_T;
  205. }
  206. ///////////////////// OLD LOAD CODE //////////////////////////
  207. void g1_map_cell_class::load_v2(i4_file_class *fp, w16 *tile_remap)
  208. {
  209. object_list=0;
  210. // function_block=0xffff;
  211. type=tile_remap[fp->read_16()];
  212. flags=fp->read_8();
  213. fp->read_8();
  214. fp->read_16(); // nearest node gone
  215. top_left_normal=0x8000;
  216. bottom_right_normal=0x8000;
  217. }
  218. void g1_map_cell_class::load_v3(i4_file_class *fp, w16 *tile_remap)
  219. {
  220. object_list=0;
  221. type=tile_remap[fp->read_16()];
  222. flags=fp->read_16();
  223. fp->read_8(); // height not used anymore
  224. fp->read_16(); // nearest node gone
  225. top_left_normal=0x8000;
  226. bottom_right_normal=0x8000;
  227. }
  228. void g1_map_cell_class::load_v4(i4_file_class *fp, w16 *tile_remap)
  229. {
  230. object_list=0;
  231. type=tile_remap[fp->read_16()];
  232. flags=fp->read_16();
  233. fp->read_16(); // nearest node gone
  234. if (type==0)
  235. flags |= IS_GROUND;
  236. top_left_normal=0x8000;
  237. bottom_right_normal=0x8000;
  238. }
  239. void g1_map_cell_class::load_v5(i4_file_class *fp, w16 *tile_remap)
  240. {
  241. object_list=0;
  242. type = tile_remap[fp->read_16()];
  243. fp->read_8();
  244. flags=fp->read_16();
  245. fp->read_16(); // nearest_node gone
  246. if (type==0)
  247. flags |= IS_GROUND;
  248. top_left_normal=0x8000;
  249. bottom_right_normal=0x8000;
  250. }
  251. void g1_map_cell_class::load_v6(i4_file_class *fp, w16 *tile_remap)
  252. {
  253. object_list=0;
  254. type = tile_remap[fp->read_16()];
  255. fp->read_8();
  256. w32 blah = fp->read_32(); // height info not used anymoe
  257. if (blah)
  258. {
  259. fp->read_8();
  260. fp->read_16();
  261. fp->read_float();
  262. fp->read_float();
  263. }
  264. flags=fp->read_16();
  265. fp->read_16(); // nearest_node gone
  266. if (type==0)
  267. flags |= IS_GROUND;
  268. top_left_normal=0x8000;
  269. bottom_right_normal=0x8000;
  270. }
  271. void g1_map_cell_class::unfog(int x, int y)
  272. {
  273. flags&=~FOGGED;
  274. g1_map_vertex_class *v=g1_verts+x+y*(g1_map_width+1);
  275. v->set_flag(g1_map_vertex_class::FOGGED,0);
  276. v++;
  277. v->set_flag(g1_map_vertex_class::FOGGED,0);
  278. v+=g1_map_width;
  279. v->set_flag(g1_map_vertex_class::FOGGED,0);
  280. v++;
  281. v->set_flag(g1_map_vertex_class::FOGGED,0);
  282. }