obj3d.cc 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  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 <string.h>
  9. #include "obj3d.hh"
  10. #include "loaders/dir_load.hh"
  11. #include "checksum/checksum.hh"
  12. #include "string/str_checksum.hh"
  13. #include "saver_id.hh"
  14. #include "time/profile.hh"
  15. #ifdef i4_NEW_CHECK
  16. #undef new
  17. #endif
  18. void g1_quad_object_class::scale(i4_float value)
  19. {
  20. for (int j=0; j<num_animations; j++)
  21. {
  22. g1_vert_class *p=animation[j].vertex;
  23. for (int i=0; i<animation[j].num_frames; i++)
  24. {
  25. g1_vert_class *p=get_verts(j,i);
  26. for (int k=0; k<num_vertex; k++)
  27. p[k].v*=value;
  28. }
  29. }
  30. for (w32 a=0; a<num_quad; a++)
  31. quad[a].texture_scale *= value;
  32. for (w32 m=0; m<num_mounts; m++)
  33. mount[m] *= value;
  34. extent *= value;
  35. }
  36. void g1_quad_object_class::translate(i4_float xadd, i4_float yadd, i4_float zadd)
  37. {
  38. for (int j=0; j<num_animations; j++)
  39. {
  40. g1_vert_class *p=animation[j].vertex;
  41. for (int i=0; i<animation[j].num_frames; i++)
  42. {
  43. g1_vert_class *p=get_verts(j,i);
  44. for (int k=0; k<num_vertex; k++)
  45. p[k].v += i4_3d_point_class(xadd, yadd, zadd);
  46. }
  47. }
  48. }
  49. void g1_quad_object_class::calc_extents()
  50. {
  51. int i,j,k;
  52. i4_float minx, maxx, miny, maxy, minz, maxz;
  53. extent=0.0;
  54. minx=miny=minz=9999;
  55. maxx=maxy=maxz=-9999;
  56. for(i=0;i<num_animations;i++)
  57. {
  58. for(j=0;j<animation[i].num_frames;j++)
  59. {
  60. for(k=j*num_vertex;k<(j*num_vertex)+num_vertex;k++)
  61. {
  62. if(animation[i].vertex[k].v.length()>extent) extent=animation[i].vertex[k].v.length();
  63. if(animation[i].vertex[k].v.x < minx) minx=animation[i].vertex[k].v.x;
  64. if(animation[i].vertex[k].v.y < miny) miny=animation[i].vertex[k].v.y;
  65. if(animation[i].vertex[k].v.z < minz) minz=animation[i].vertex[k].v.z;
  66. if(animation[i].vertex[k].v.x > maxx) maxx=animation[i].vertex[k].v.x;
  67. if(animation[i].vertex[k].v.y > maxy) maxy=animation[i].vertex[k].v.y;
  68. if(animation[i].vertex[k].v.z > maxz) maxz=animation[i].vertex[k].v.z;
  69. }
  70. }
  71. }
  72. }
  73. void g1_quad_object_class::update(i4_float time)
  74. {
  75. for (int i=0; i<num_special; i++)
  76. {
  77. g1_texture_animation *t = &special[i];
  78. g1_quad_class *q = &quad[t->quad_number];
  79. int max_frames = t->max_frames;
  80. if (max_frames==0)
  81. {
  82. // panning
  83. i4_float fr, du,dv;
  84. fr = time*t->speed;
  85. fr -= i4_float(int(fr));
  86. du = t->du*fr;
  87. dv = t->dv*fr;
  88. for (int n=0; n<q->num_verts(); n++)
  89. {
  90. q->u[n] = t->u[n] + du;
  91. q->v[n] = t->v[n] + dv;
  92. }
  93. }
  94. else
  95. {
  96. // animation
  97. int frame = int(time*t->speed)%max_frames;
  98. int x = frame%t->frames_x;
  99. int y = frame/t->frames_x;
  100. for (int n=0; n<q->num_verts(); n++)
  101. {
  102. q->u[n] = t->u[n] + t->du*x;
  103. q->v[n] = t->v[n] + t->dv*y;
  104. }
  105. }
  106. }
  107. }
  108. i4_bool g1_quad_object_class::intersect(const i4_3d_vector &point,
  109. const i4_3d_vector &ray,
  110. int anim, int frame,
  111. i4_float *intersect_t,
  112. int *hit_poly,
  113. i4_3d_vector *normal)
  114. {
  115. g1_vert_class *verts = get_verts(anim, frame);
  116. g1_quad_class *q = quad;
  117. i4_bool hit = i4_F;
  118. i4_float new_t = 1.0;
  119. for (int i=0; i<num_quad; i++, q++)
  120. {
  121. if (q->normal.dot(ray)>=0)
  122. // wrong direction, next quad
  123. continue;
  124. // find intersection point in polygon's plane
  125. i4_float d = q->normal.dot(verts[q->vertex_ref[0]].v);
  126. i4_float num = d - q->normal.dot(point);
  127. i4_float den = q->normal.dot(ray);
  128. i4_float t = num/den;
  129. if (t<0)
  130. // behind the observer, next
  131. continue;
  132. if (t>=new_t)
  133. // check if new intersection is closer than previous ones
  134. continue;
  135. // project point into plane
  136. int j, u,v;
  137. i4_3d_vector new_ray(ray);
  138. new_ray *= t;
  139. i4_3d_vector new_point(point);
  140. new_point += new_ray;
  141. // determine orthogonal plane for 2D point in polygon test
  142. i4_3d_vector abs_normal;
  143. abs_normal.set(fabs(q->normal.x), fabs(q->normal.y), fabs(q->normal.z));
  144. if (abs_normal.x>abs_normal.y && abs_normal.x>abs_normal.z)
  145. {
  146. // test in yz plane
  147. u = 1; v = 2;
  148. }
  149. else if (abs_normal.y>abs_normal.x && abs_normal.y>abs_normal.z)
  150. {
  151. // test in zx plane
  152. u = 2; v = 0;
  153. }
  154. else
  155. {
  156. // test in xy plane
  157. u = 0; v = 1;
  158. }
  159. // count first axis crossings of polygon to determine whether point in polygon
  160. i4_float
  161. dx1, dy1,
  162. dx2 = verts[q->vertex_ref[0]].v[u] - new_point[u],
  163. dy2 = verts[q->vertex_ref[0]].v[v] - new_point[v];
  164. int
  165. x_sign1, y_sign1,
  166. x_sign2 = dx2>0,
  167. y_sign2 = dy2>0;
  168. int inside=0;
  169. for (j=q->num_verts()-1; j>=0; j--)
  170. {
  171. dx1 = verts[q->vertex_ref[j]].v[u] - new_point[u];
  172. dy1 = verts[q->vertex_ref[j]].v[v] - new_point[v];
  173. x_sign1 = dx1>0;
  174. y_sign1 = dy1>0;
  175. if (y_sign1 != y_sign2)
  176. {
  177. // crosses x-axis
  178. if (x_sign1 != x_sign2)
  179. {
  180. // check if crossing point is positive ((slope2>slope1) ^ (y2>y1))
  181. if ((dx2*dy1 > dx1*dy2) ^ (dy2>dy1))
  182. inside = !inside;
  183. }
  184. else if (x_sign1)
  185. // crosses +x-axis
  186. inside = !inside;
  187. y_sign2 = y_sign1;
  188. }
  189. x_sign2 = x_sign1;
  190. dx2 = dx1;
  191. dy2 = dy1;
  192. }
  193. if (inside)
  194. {
  195. // update current intersection
  196. new_t = t;
  197. if (hit_poly) *hit_poly=i;
  198. if (normal) *normal=q->normal;
  199. hit = i4_T;
  200. }
  201. }
  202. if (intersect_t) *intersect_t=new_t;
  203. return hit;
  204. }
  205. i4_bool g1_quad_object_class::get_mount_point(char *name, i4_3d_vector& vect) const
  206. {
  207. w32 id = i4_check_sum32(name, strlen(name));
  208. for (int i=0; i<num_mounts; i++)
  209. if (mount_id[i] == id)
  210. {
  211. vect = mount[i];
  212. return i4_T;
  213. }
  214. return i4_F;
  215. }
  216. static i4_profile_class pf_load_tnames("models:load_tname");
  217. static i4_profile_class pf_load_quads("models:load_quads");
  218. static i4_profile_class pf_load_verts("models:load_verts");
  219. static i4_profile_class pf_load_mounts("models:load_mounts");
  220. static i4_profile_class pf_load_tanim("models:load_tanim");
  221. g1_quad_object_class *g1_base_object_loader_class::load(i4_loader_class *fp)
  222. {
  223. int i,j,k;
  224. if (!fp)
  225. return 0;
  226. if (!fp->goto_section(G1_SECTION_MODEL_TEXTURE_NAMES))
  227. return 0;
  228. obj=allocate_object();
  229. pf_load_tnames.start();
  230. w16 quads=fp->read_16();
  231. set_num_quads(quads);
  232. char name[256];
  233. for (i=0; i<quads; i++)
  234. {
  235. w16 len = fp->read_16();
  236. fp->read(name, len);
  237. name[len]=0;
  238. store_texture_name(i, name);
  239. }
  240. pf_load_tnames.stop();
  241. if (!fp->goto_section(G1_SECTION_MODEL_QUADS))
  242. return 0;
  243. pf_load_quads.start();
  244. fp->read_16(); // repeat of num quads (why?)
  245. for (i=0; i<quads; i++)
  246. {
  247. w16 ref[4];
  248. i4_float u[4],v[4], scale;
  249. i4_3d_vector normal;
  250. w32 flags;
  251. for (j=0; j<4; ++j)
  252. {
  253. ref[j]=fp->read_16();
  254. u[j]=fp->read_float();
  255. v[j]=fp->read_float();
  256. }
  257. scale=fp->read_float();
  258. flags=fp->read_16();
  259. normal.x=fp->read_float();
  260. normal.y=fp->read_float();
  261. normal.z=fp->read_float();
  262. create_quad(i, 4, ref, flags);
  263. store_texture_params(i, scale, u, v);
  264. store_quad_normal(i, normal);
  265. }
  266. pf_load_quads.stop();
  267. if (!fp->goto_section(G1_SECTION_MODEL_VERT_ANIMATION))
  268. return 0;
  269. pf_load_verts.start();
  270. w16 nv=fp->read_16(); // number of vertexes in object
  271. w16 na=fp->read_16(); // number of animations
  272. set_num_vertex(nv);
  273. set_num_animations(na);
  274. for (i=0; i<na; i++)
  275. {
  276. int len=fp->read_16();
  277. fp->read(name, len);
  278. name[len]=0;
  279. w16 frames = fp->read_16();
  280. create_animation(i, name, frames);
  281. for (j=0; j<frames; j++)
  282. {
  283. for (k=0; k<nv; k++)
  284. {
  285. i4_3d_vector v,n;
  286. v.x=fp->read_float();
  287. v.y=fp->read_float();
  288. v.z=fp->read_float();
  289. n.x=fp->read_float();
  290. n.y=fp->read_float();
  291. n.z=fp->read_float();
  292. create_vertex(i,j,k,v);
  293. store_vertex_normal(i,j,k,n);
  294. }
  295. }
  296. }
  297. pf_load_verts.stop();
  298. // optional mount points
  299. if (fp->goto_section("GMOD Mounts"))
  300. {
  301. pf_load_mounts.start();
  302. w16 mounts = fp->read_16();
  303. set_num_mount_points(mounts);
  304. for (i=0; i<mounts; i++)
  305. {
  306. i4_3d_vector v;
  307. int len=fp->read_16();
  308. fp->read(name, len);
  309. name[len]=0;
  310. v.x = fp->read_float();
  311. v.y = fp->read_float();
  312. v.z = fp->read_float();
  313. create_mount_point(i,name,v);
  314. }
  315. pf_load_mounts.stop();
  316. }
  317. // optional mount points
  318. if (fp->goto_section("GMOD Texture Animations"))
  319. {
  320. pf_load_tanim.start();
  321. w16 anims = fp->read_16();
  322. set_num_texture_animations(anims);
  323. for (i=0; i<anims; i++)
  324. {
  325. w16 q;
  326. w8 maxf, frx;
  327. i4_float sp,du,dv;
  328. fp->read_format("211fff",&q,&maxf,&frx,&sp,&du,&dv);
  329. if (maxf>0)
  330. create_texture_animation(i,q,maxf,frx,du,dv,sp);
  331. else
  332. create_texture_pan(i,q,du,dv,sp);
  333. }
  334. pf_load_tanim.stop();
  335. }
  336. finish_object();
  337. return obj;
  338. }