crate.cc 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  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 "objs/crate.hh"
  9. #include "objs/model_draw.hh"
  10. #include "lisp/li_class.hh"
  11. #include "objs/model_id.hh"
  12. #include "object_definer.hh"
  13. #include "map.hh"
  14. #include "map_man.hh"
  15. #include "g1_render.hh"
  16. #include "math/pi.hh"
  17. #include "r1_api.hh"
  18. #include "objs/stank.hh"
  19. #include "player.hh"
  20. #include "resources.hh"
  21. static li_symbol_class_member type("type"), amount("amount");
  22. static li_symbol_ref li_health("health"), li_missile("missile"),
  23. li_bullet("bullet"), li_money("money"), li_small("small"), li_large("large"),
  24. li_chain("chain_gun");
  25. static g1_model_ref
  26. health_model("crate_health"),
  27. missile_model("crate_missiles"),
  28. money_model("crate_money"),
  29. chain_model("crate_minigun"),
  30. bullet_model("crate_bullets"),
  31. health_flare_model("powerup_flare_blue"),
  32. missile_flare_model("powerup_flare_white"),
  33. money_flare_model("powerup_flare_green"),
  34. bullet_flare_model("powerup_flare_red"),
  35. chain_flare_model("powerup_flare_yellow");
  36. static li_int_class_member li_ticks_left("ticks_left");
  37. static li_float_class_member li_yvel("yvel");
  38. S1_SFX(bullet_sfx, "misc/main_barrel_powerup_22khz.wav", 0, 200);
  39. S1_SFX(health_sfx, "misc/health_powerup_three_22khz.wav", 0, 200);
  40. S1_SFX(missile_sfx, "misc/missle_powerup_22khz.wav", 0, 200);
  41. S1_SFX(chain_sfx, "misc/supertank_chain_gun_refuel.wav", 0, 200);
  42. S1_SFX(money_sfx, "misc/powerup_money_22khz.wav", 0, 200);
  43. float &g1_crate_class::yvel() { return vars->get(li_yvel); }
  44. int &g1_crate_class::ticks_left() { return vars->get(li_ticks_left); }
  45. g1_object_definer<g1_crate_class>
  46. g1_create_def("crate", g1_object_definition_class::EDITOR_SELECTABLE);
  47. i4_bool g1_crate_class::occupy_location()
  48. {
  49. h=lh=g1_get_map()->terrain_height(x,y)+float_height();
  50. return g1_object_class::occupy_location_corners();
  51. }
  52. int g1_crate_class::added_money()
  53. {
  54. return get_amount()==SMALL ?
  55. g1_resources.small_money_added :
  56. g1_resources.large_money_added;
  57. }
  58. int g1_crate_class::added_bullets()
  59. {
  60. return get_amount()==SMALL ?
  61. g1_resources.small_bullets_added :
  62. g1_resources.large_bullets_added;
  63. }
  64. int g1_crate_class::added_health()
  65. {
  66. return get_amount()==SMALL ?
  67. g1_resources.small_health_added :
  68. g1_resources.large_health_added;
  69. }
  70. int g1_crate_class::added_missiles()
  71. {
  72. return get_amount()==SMALL ?
  73. g1_resources.small_missiles_added :
  74. g1_resources.large_missiles_added;
  75. }
  76. int g1_crate_class::added_chain()
  77. {
  78. return get_amount()==SMALL ?
  79. g1_resources.small_chain_added :
  80. g1_resources.large_chain_added;
  81. }
  82. g1_crate_class::ctype g1_crate_class::get_type()
  83. {
  84. li_symbol *s=vars->get(type);
  85. if (s==li_health.get())
  86. return HEALTH;
  87. else if (s==li_missile.get())
  88. return MISSILE;
  89. else if (s==li_bullet.get())
  90. return BULLET;
  91. else if (s==li_chain.get())
  92. return CHAIN;
  93. else return MONEY;
  94. }
  95. void g1_crate_class::set_type(ctype x)
  96. {
  97. li_symbol *s;
  98. int sub_type;
  99. switch (x)
  100. {
  101. case HEALTH :
  102. {
  103. s=li_health.get();
  104. draw_params.setup(health_model.get());
  105. sub_type=health_flare_model.value;
  106. } break;
  107. case MISSILE :
  108. {
  109. s=li_missile.get();
  110. draw_params.setup(missile_model.get());
  111. sub_type=missile_flare_model.value;
  112. } break;
  113. case BULLET :
  114. {
  115. s=li_bullet.get();
  116. draw_params.setup(bullet_model.get());
  117. sub_type=bullet_flare_model.value;
  118. } break;
  119. case CHAIN :
  120. {
  121. s=li_chain.get();
  122. draw_params.setup(chain_model.get());
  123. sub_type=chain_flare_model.value;
  124. } break;
  125. case MONEY :
  126. {
  127. s=li_money.get();
  128. draw_params.setup(money_model.get());
  129. sub_type=money_flare_model.value;
  130. } break;
  131. }
  132. // if (get_amount()==LARGE)
  133. // num_mini_objects=3;
  134. // else
  135. // num_mini_objects=3;
  136. // for (int i=0; i<num_mini_objects; i++)
  137. // mini_objects[i].defmodeltype=sub_type;
  138. vars->set(type, s);
  139. }
  140. g1_crate_class::atype g1_crate_class::get_amount()
  141. {
  142. if (vars->get(amount)==li_small.get())
  143. return SMALL;
  144. else
  145. return LARGE;
  146. }
  147. void g1_crate_class::set_amount(atype x)
  148. {
  149. switch (x)
  150. {
  151. case SMALL :
  152. vars->set(amount, li_small.get());
  153. break;
  154. case LARGE :
  155. vars->set(amount, li_large.get());
  156. break;
  157. }
  158. }
  159. void g1_crate_class::think()
  160. {
  161. if (ticks_to_think || ticks_left())
  162. {
  163. if (ticks_left()>0)
  164. {
  165. ticks_left()--;
  166. if (ticks_left()==0)
  167. {
  168. unoccupy_location();
  169. request_remove();
  170. return ;
  171. }
  172. }
  173. else
  174. ticks_to_think--;
  175. request_think();
  176. theta+=0.1;
  177. pitch+=0.01;
  178. // mini_objects[0].rotation.x-=0.2;
  179. // mini_objects[0].rotation.y-=0;
  180. // mini_objects[0].rotation.z-=0;
  181. // mini_objects[1].rotation.x-=0;
  182. // mini_objects[1].rotation.y-=0.2;
  183. // mini_objects[1].rotation.z-=0;
  184. // mini_objects[2].rotation.x-=0.2;
  185. // mini_objects[2].rotation.y-=0.15;
  186. // mini_objects[2].rotation.z-=0.0;
  187. // float mh=g1_get_map()->map_height(x,y,h);
  188. // if (mh!=h+float_height())
  189. // {
  190. // h+=yvel();
  191. // yvel()-=g1_resources.gravity;
  192. // if (h+float_height()<mh)
  193. // {
  194. // h=mh+float_height();
  195. // yvel()=0;
  196. // request_remove();
  197. // }
  198. // }
  199. }
  200. }
  201. void g1_crate_class::go_away()
  202. {
  203. unoccupy_location();
  204. request_remove();
  205. }
  206. void g1_crate_class::note_stank_near(g1_player_piece_class *s)
  207. {
  208. float dist_sqrd=(s->x-x)*(s->x-x)+(s->y-y)*(s->y-y)+(s->h-h)*(s->h-h);
  209. if (dist_sqrd<0.5*0.5)
  210. {
  211. char msg[100];
  212. switch (get_type())
  213. {
  214. case HEALTH :
  215. if (s->health!=s->ammo[3].amount)
  216. {
  217. int old=s->health;
  218. s->health+=added_health();
  219. if (s->health>s->ammo[3].amount)
  220. s->health=s->ammo[3].amount;
  221. go_away();
  222. sprintf(msg, "+%d Armor", s->health-old);
  223. health_sfx.play();
  224. }
  225. break;
  226. case CHAIN :
  227. if (s->ammo[2].amount!=s->ammo[2].ammo_type->max_amount)
  228. {
  229. int old=s->ammo[2].amount;
  230. s->ammo[2].amount+=added_bullets();
  231. if (s->ammo[2].amount>s->ammo[2].ammo_type->max_amount)
  232. s->ammo[2].amount=s->ammo[2].ammo_type->max_amount;
  233. go_away();
  234. sprintf(msg, "+%d Mini Gun", s->ammo[2].amount-old);
  235. }
  236. break;
  237. case BULLET :
  238. if (s->ammo[0].amount!=s->ammo[0].ammo_type->max_amount)
  239. {
  240. int old=s->ammo[0].amount;
  241. s->ammo[0].amount+=added_chain();
  242. if (s->ammo[0].amount>s->ammo[0].ammo_type->max_amount)
  243. s->ammo[0].amount=s->ammo[0].ammo_type->max_amount;
  244. go_away();
  245. sprintf(msg, "+%d Main Rounds", s->ammo[0].amount-old);
  246. chain_sfx.play();
  247. }
  248. break;
  249. case MISSILE :
  250. if (s->ammo[1].amount!=s->ammo[1].ammo_type->max_amount)
  251. {
  252. int old=s->ammo[1].amount;
  253. s->ammo[1].amount+=added_missiles();
  254. if (s->ammo[1].amount>s->ammo[1].ammo_type->max_amount)
  255. s->ammo[1].amount=s->ammo[1].ammo_type->max_amount;
  256. go_away();
  257. sprintf(msg, "+%d Missiles", s->ammo[1].amount-old);
  258. missile_sfx.play();
  259. }
  260. break;
  261. case MONEY :
  262. g1_player_man.get(s->player_num)->money()+=added_money();
  263. go_away();
  264. sprintf(msg, "+ $%d ", added_money());
  265. money_sfx.play();
  266. break;
  267. }
  268. g1_player_man.show_message(msg, 0x00ff00, s->player_num);
  269. }
  270. }
  271. void g1_crate_class::draw(g1_draw_context_class *context)
  272. {
  273. i4_transform_class *old = context->transform;
  274. i4_transform_class view_transform;
  275. context->transform = &view_transform;
  276. i4_3d_vector my_interp_position = world_transform->t;
  277. view_transform.multiply(*old,*(world_transform));
  278. // draw without lighting
  279. g1_render.render_object(draw_params.model,
  280. &view_transform,
  281. 0,
  282. 1,
  283. player_num,
  284. 0,
  285. 0,
  286. 0);
  287. // for (int i=0; i<num_mini_objects; i++)
  288. // {
  289. // g1_quad_object_class *model = g1_model_list_man.get_model(mini_objects[i].defmodeltype);
  290. // i4_transform_class l2w, w2v, tmp, l2v;
  291. // mini_objects[i].calc_transform(g1_render.frame_ratio, &l2w);
  292. // tmp.translate(x,y,h);
  293. // tmp.multiply(l2w);
  294. // l2v.multiply(*old, tmp);
  295. // g1_render.render_object(model,
  296. // &l2v,
  297. // 0,
  298. // 1,
  299. // player_num,
  300. // 0,
  301. // 0,
  302. // 0);
  303. // }
  304. ticks_to_think=10;
  305. request_think();
  306. context->transform=old;
  307. }
  308. g1_crate_class::g1_crate_class(g1_object_type id, g1_loader_class *fp)
  309. : g1_object_class(id, fp)
  310. {
  311. ticks_to_think=0;
  312. // allocate_mini_objects(3,"crate mini objects");
  313. // g1_mini_object *a=mini_objects;
  314. // int i;
  315. // for (i=0; i<3; i++)
  316. // {
  317. // a->x=0; a->y=0; a->h=0;
  318. // a->offset = i4_3d_vector(0,0,0);
  319. // a++;
  320. // }
  321. set_type(get_type()); // to setup the draw params
  322. // mini_objects[1].rotation.y=i4_pi()/2;
  323. // mini_objects[2].rotation.z=i4_pi()/2;
  324. // for (i=0; i<3; i++)
  325. // mini_objects[i].grab_old();
  326. }
  327. void g1_crate_class::setup(const i4_3d_vector &pos, ctype t, atype amount, int ticks)
  328. {
  329. ticks_left()=ticks;
  330. lx=x=pos.x;
  331. ly=y=pos.y;
  332. lh=h=pos.z;
  333. set_type(t);
  334. set_amount(amount);
  335. if (!get_flag(MAP_OCCUPIED))
  336. {
  337. if (!occupy_location())
  338. return;
  339. }
  340. }
  341. void g1_crate_class::object_changed_by_editor(g1_object_class *who, li_class *old_values)
  342. {
  343. if (who==this)
  344. set_type(get_type());
  345. }