player.cc 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  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 "player.hh"
  9. #include "team_api.hh"
  10. #include "error/error.hh"
  11. #include "g1_limits.hh"
  12. #include "image/color.hh"
  13. #include "map.hh"
  14. #include "g1_object.hh"
  15. #include "objs/map_piece.hh"
  16. #include "objs/stank.hh"
  17. #include "palette/pal.hh"
  18. #include "saver.hh"
  19. #include "border_frame.hh"
  20. #include "map_man.hh"
  21. #include "dll/dll_man.hh"
  22. #include "time/profile.hh"
  23. #include "lisp/li_class.hh"
  24. #include "lisp/li_load.hh"
  25. #include "li_objref.hh"
  26. #include "lisp/li_init.hh"
  27. #include "objs/path_object.hh"
  28. #include "lisp/li_dialog.hh"
  29. #include "controller.hh"
  30. #include "g1_tint.hh"
  31. enum
  32. {
  33. DATA_VERSION1=1,
  34. DATA_VERSION2,
  35. DATA_VERSION3,
  36. DATA_VERSION // added ai saving
  37. };
  38. g1_player_manager_class g1_player_man;
  39. static li_symbol_ref li_upgrade_kill_ratio("upgrade_kill_ratio");
  40. static li_int_class_member li_money("money"), li_income_rate("income_rate"),
  41. li_num_points("num_points"), li_num_stank_deaths("num_stank_deaths"),
  42. li_num_stank_lives("num_stank_lives");
  43. static li_float_class_member li_damage_multiplier("damage_multiplier");
  44. static li_g1_ref_class_member current_taveover_target("current_taveover_target");
  45. static li_string_class_member li_team_flag("team_flag");
  46. static li_symbol_ref stank("stank");
  47. int &g1_player_info_class::num_points() { return get_vars()->get(li_num_points); }
  48. int &g1_player_info_class::num_stank_deaths() { return get_vars()->get(li_num_stank_deaths); }
  49. int &g1_player_info_class::num_stank_lives() { return get_vars()->get(li_num_stank_lives); }
  50. int &g1_player_info_class::money() { return get_vars()->get(li_money); }
  51. int &g1_player_info_class::income_rate() { return get_vars()->get(li_income_rate); }
  52. float &g1_player_info_class::damage_multiplier() { return get_vars()->get(li_damage_multiplier);}
  53. void g1_player_manager_class::show_message(const i4_const_str &message, w32 color, int player_for)
  54. {
  55. if (player_for==local_player && g1_current_controller.get())
  56. g1_current_controller->scroll_message(message, color);
  57. }
  58. g1_player_info_class::g1_player_info_class()
  59. : ai(0),
  60. owned_objects(0,32)
  61. {
  62. refresh_money=1;
  63. }
  64. void g1_player_info_class::remove_object(w32 global_id)
  65. {
  66. if (ai) ai->object_lost(global_id);
  67. g1_object_class *o=g1_global_id.checked_get(global_id);
  68. if (o==commander.get())
  69. set_commander(0);
  70. int t=owned_objects.size();
  71. for (int i=0; i<t; i++)
  72. if (owned_objects[i] == global_id)
  73. {
  74. owned_objects[i]=g1_global_id.invalid_id();
  75. return ;
  76. }
  77. i4_warning("object not found");
  78. }
  79. void g1_player_info_class::add_object(w32 global_id)
  80. {
  81. if (ai)
  82. ai->object_added(global_id);
  83. g1_object_class *o=g1_global_id.get(global_id);
  84. if (!commander.get() && o->id==g1_get_object_type(stank.get()))
  85. set_commander(g1_player_piece_class::cast(o));
  86. int t=owned_objects.size();
  87. for (int i=0; i<t; i++)
  88. if (!g1_global_id.check_id(owned_objects[i]))
  89. {
  90. owned_objects[i]=global_id;
  91. return;
  92. }
  93. owned_objects.add(global_id);
  94. }
  95. float g1_player_info_class::kill_ratio()
  96. {
  97. int d=num_stank_deaths();
  98. if (!d) d=1;
  99. return num_points() / (float)d;
  100. }
  101. void g1_player_info_class::calc_upgrade_level()
  102. {
  103. supertank_upgrade_level=0;
  104. // run through the list "upgrage_time" in player_vars which has times to
  105. // upgrade
  106. int points=i4_f_to_i(kill_ratio());
  107. li_object *l = li_get_value(li_upgrade_kill_ratio.get());
  108. while (l)
  109. {
  110. int required_number = li_int::get(li_car(l,0),0)->value();
  111. if (required_number<=points) // do we have enough kills for this level
  112. supertank_upgrade_level++;
  113. l = li_cdr(l,0); // next item in list
  114. }
  115. if (supertank_upgrade_level<0)
  116. supertank_upgrade_level=0;
  117. }
  118. static w32 convert_color_to_player(w32 player_number, i4_color color,
  119. const i4_pixel_format *fmt)
  120. {
  121. sw32 r,g,b;
  122. r=(color>>16)&0xff;
  123. g=(color>>8)&0xff;
  124. b=(color>>0)&0xff;
  125. r=i4_f_to_i(g1_player_tint_data[player_number].r * r);
  126. if (r>255) r=255;
  127. g=i4_f_to_i(g1_player_tint_data[player_number].g * g);
  128. if (g>255) g=255;
  129. b=i4_f_to_i(g1_player_tint_data[player_number].b * b);
  130. if (b>255) b=255;
  131. return (r<<16)|(g<<8)|b;
  132. }
  133. g1_player_manager_class::g1_player_manager_class()
  134. {
  135. }
  136. void g1_player_info_class::init_colors(w32 player_num, const i4_pixel_format *fmt)
  137. {
  138. map_player_building=convert_color_to_player(player_num, 0x404040, fmt);
  139. map_player_color=convert_color_to_player(player_num, 0xffffff, fmt);
  140. map_attacking_player_color=convert_color_to_player(player_num, 0xdcdcdc ,fmt);
  141. map_select_player_color=(200<<16) | (200<<8) | 0;
  142. }
  143. void g1_player_info_class::new_game(g1_player_type num)
  144. {
  145. player_num = num;
  146. supertank_upgrade_level = 0;
  147. if (num==1)
  148. team=G1_ALLY;
  149. else
  150. team=G1_ENEMY;
  151. }
  152. void g1_player_info_class::set_ai(g1_team_api_class *_ai)
  153. {
  154. if (ai)
  155. delete ai;
  156. ai = _ai;
  157. if (ai)
  158. ai->player = this;
  159. }
  160. i4_profile_class pf_player_info_class_ai_think("player_info:: ai->think()");
  161. void g1_player_info_class::add_points(int points)
  162. {
  163. num_points()+=points;
  164. calc_upgrade_level();
  165. }
  166. void g1_player_info_class::supertank_died(g1_object_class *guy_who_killed_him)
  167. {
  168. num_stank_deaths()++;
  169. calc_upgrade_level();
  170. }
  171. void g1_player_info_class::think()
  172. {
  173. li_class_context vars_context(li_class::get(vars.get(),0));
  174. pf_player_info_class_ai_think.start();
  175. if (ai)
  176. {
  177. ai->think();
  178. ai->post_think();
  179. }
  180. pf_player_info_class_ai_think.stop();
  181. li_money() += li_income_rate();
  182. if (li_money() > 99999)
  183. li_money() = 99999;
  184. else if (li_money() < 0)
  185. li_money() = 0;
  186. if (g1_border.get())
  187. g1_border->update();
  188. }
  189. i4_bool g1_player_info_class::load(g1_loader_class *fp)
  190. {
  191. w16 ver=0,data_size;
  192. i4_bool ret=i4_F;
  193. vars=0;
  194. continue_wait=i4_F;
  195. if (fp)
  196. fp->get_version(ver, data_size);
  197. if (!ai) i4_error("no ai in player_info load");
  198. ai->init();
  199. ret = i4_T;
  200. switch (ver)
  201. {
  202. case DATA_VERSION:
  203. vars=li_class::get(li_load_typed_object("player_vars", fp, fp->li_remap,0),0);
  204. ai->load(fp);
  205. break;
  206. case DATA_VERSION3:
  207. vars=li_class::get(li_load_typed_object("player_vars", fp, fp->li_remap,0),0);
  208. break;
  209. default:
  210. if (fp)
  211. fp->seek(fp->tell() + data_size);
  212. ret = i4_F;
  213. break;
  214. }
  215. if (!vars.get())
  216. vars=li_class::get(li_new("player_vars"), 0);
  217. li_class_context c(get_vars());
  218. team_flag.set_name(li_team_flag());
  219. return ret;
  220. }
  221. void g1_player_info_class::save(g1_saver_class *fp)
  222. {
  223. fp->start_version(DATA_VERSION);
  224. li_save_object(fp, vars.get(), 0);
  225. ai->save(fp);
  226. fp->end_version();
  227. }
  228. //(OLI) temporary hack to allow DLL reloading of AI
  229. //{{{
  230. extern void (*g1_reload_dll)(void);
  231. void reload_ais()
  232. {
  233. i4_dll_man.load(i4_const_str("ai_joe.dll"), i4_T);
  234. }
  235. //}}}
  236. void g1_player_manager_class::set_default_ai(const char *name)
  237. {
  238. default_ai=name;
  239. if (default_ai)
  240. for (w32 i=0; i<G1_MAX_PLAYERS; i++)
  241. {
  242. if (list[i] && !list[i]->get_ai())
  243. list[i]->set_ai(g1_create_ai(default_ai));
  244. }
  245. }
  246. void g1_player_manager_class::init()
  247. {
  248. for (w32 i=0; i<G1_MAX_PLAYERS; i++)
  249. {
  250. list[i] = new g1_player_info_class();
  251. list[i]->new_game(i);
  252. }
  253. local_player=1; //(OLI) this needs to be changed to the appropriate number
  254. list[local_player]->set_ai(g1_create_ai("human"));
  255. if (default_ai)
  256. for (w32 i=0; i<G1_MAX_PLAYERS; i++)
  257. {
  258. if (list[i] && !list[i]->get_ai())
  259. list[i]->set_ai(g1_create_ai(default_ai));
  260. }
  261. //(OLI)
  262. g1_reload_dll = reload_ais;
  263. }
  264. void g1_player_manager_class::uninit()
  265. {
  266. for (w32 i=0; i<G1_MAX_PLAYERS; i++)
  267. {
  268. if (list[i])
  269. {
  270. list[i]->set_ai(0);
  271. delete list[i];
  272. }
  273. list[i]=0;
  274. }
  275. }
  276. void g1_player_manager_class::unload_ai(g1_team_api_definition_class *definer)
  277. {
  278. for (int i=0; i<G1_MAX_PLAYERS; i++)
  279. if (g1_player_man.get(i) &&
  280. g1_player_man.get(i)->get_ai() &&
  281. g1_player_man.get(i)->get_ai()->definer() == definer)
  282. g1_player_man.get(i)->set_ai(0);
  283. }
  284. void g1_player_manager_class::init_colors(const i4_pixel_format *fmt,
  285. r1_render_api_class *api)
  286. {
  287. g1_init_color_tints(api);
  288. for (w32 i=0; i<G1_MAX_PLAYERS; i++)
  289. list[i]->init_colors(i, fmt);
  290. }
  291. li_object *g1_set_default_ai(li_object *o, li_environment *env)
  292. {
  293. g1_player_man.set_default_ai(li_string::get(li_car(o,0),0)->value());
  294. return 0;
  295. }
  296. li_automatic_add_function(g1_set_default_ai, "set_default_ai");