79c30836ef50df3c08567a30d6eb47c0c62da093
[supertux.git] / src / statistics.cpp
1 //
2 //  SuperTux -  A Jump'n Run
3 //  Copyright (C) 2004 Ricardo Cruz <rick2@aeiou.pt>
4 //
5 //  This program is free software; you can redistribute it and/or
6 //  modify it under the terms of the GNU General Public License
7 //  as published by the Free Software Foundation; either version 2
8 //  of the License, or (at your option) any later version.
9 //
10 //  This program is distributed in the hope that it will be useful,
11 //  but WITHOUT ANY WARRANTY; without even the implied warranty of
12 //  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 //  GNU General Public License for more details.
14 // 
15 //  You should have received a copy of the GNU General Public License
16 //  along with this program; if not, write to the Free Software
17 //  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
18 //  02111-1307, USA.
19
20 #include "utils/lispreader.h"
21 #include "utils/lispwriter.h"
22 #include "statistics.h"
23 #include "video/drawing_context.h"
24 #include "app/gettext.h"
25 #include "app/globals.h"
26 #include "resources.h"
27
28 Statistics global_stats;
29
30 std::string
31 stat_name_to_string(int stat_enum)
32 {
33   switch(stat_enum)
34     {
35     case SCORE_STAT:
36       return "score";
37     case BADGUYS_SQUISHED_STAT:
38       return "badguys-squished";
39     case SHOTS_STAT:
40       return "shots";
41     case TIME_NEEDED_STAT:
42       return "time-needed";
43     case JUMPS_STAT:
44       return "jumps";
45     }
46 }
47
48 int
49 my_min(int a, int b)
50 {
51 if(a == -1)
52   return b;
53 if(b == -1)
54   return a;
55 return std::min(a, b);
56 }
57
58 Statistics::Statistics()
59 {
60   timer.init(true);
61   display_stat = 1;
62
63   for(int i = 0; i < NUM_STATS; i++)
64     stats[i] = -1;
65 }
66
67 Statistics::~Statistics()
68 {
69 }
70
71 void
72 Statistics::parse(LispReader& reader)
73 {
74   for(int i = 0; i < NUM_STATS; i++)
75     reader.read_int(stat_name_to_string(i).c_str(), stats[i]);
76 }
77
78 void
79 Statistics::write(LispWriter& writer)
80 {
81   for(int i = 0; i < NUM_STATS; i++)
82     writer.write_int(stat_name_to_string(i), stats[i]);
83 }
84
85 #define TOTAL_DISPLAY_TIME 3400
86 #define FADING_TIME         600
87
88 void
89 Statistics::draw_worldmap_info(DrawingContext& context)
90 {
91   if(stats[SCORE_STAT] == -1)  // not initialized yet
92     return;
93
94   if(!timer.check())
95     {
96     timer.start(TOTAL_DISPLAY_TIME);
97     display_stat++;
98     if(display_stat >= NUM_STATS)
99       display_stat = 1;
100     }
101
102   int alpha;
103   if(timer.get_gone() < FADING_TIME)
104     alpha = timer.get_gone() * 255 / FADING_TIME;
105   else if(timer.get_left() < FADING_TIME)
106     alpha = timer.get_left() * 255 / FADING_TIME;
107   else
108     alpha = 255;
109
110   char str[128];
111
112   context.draw_text(white_small_text, _("Level Statistics"), Vector(550, 490), LEFT_ALLIGN, LAYER_GUI);
113
114   sprintf(str, _("Max score: %d"), stats[SCORE_STAT]);
115   context.draw_text(white_small_text, str, Vector(560, 506), LEFT_ALLIGN, LAYER_GUI);
116
117   if(display_stat == BADGUYS_SQUISHED_STAT)
118     sprintf(str, _("Max fragging: %d"), stats[BADGUYS_SQUISHED_STAT]);
119   else if(display_stat == SHOTS_STAT)
120     sprintf(str, _("Min shots: %d"), stats[SHOTS_STAT]);
121   else if(display_stat == TIME_NEEDED_STAT)
122     sprintf(str, _("Min time needed: %d"), stats[TIME_NEEDED_STAT]);
123   else// if(display_stat == JUMPS_STAT)
124     sprintf(str, _("Min jumps: %d"), stats[JUMPS_STAT]);
125
126   context.draw_text(white_small_text, str, Vector(560, 522), LAYER_GUI, LEFT_ALLIGN, NONE_EFFECT, alpha);
127 }
128
129 void
130 Statistics::draw_message_info(DrawingContext& context, std::string title)
131 {
132   if(stats[SCORE_STAT] == -1)  // not initialized yet
133     return;
134
135   context.draw_text(gold_text, title, Vector(screen->w/2, 400), CENTER_ALLIGN, LAYER_GUI);
136
137   char str[128];
138   for(int i = 0; i < NUM_STATS; i++)
139     {
140     if(i == SCORE_STAT)
141       sprintf(str, _("Max score: %d"), stats[SCORE_STAT]);
142     else if(i == BADGUYS_SQUISHED_STAT)
143       sprintf(str, _("Max fragging: %d"), stats[BADGUYS_SQUISHED_STAT]);
144     else if(i == SHOTS_STAT)
145       sprintf(str, _("Min shots: %d"), stats[SHOTS_STAT]);
146     else if(i == TIME_NEEDED_STAT)
147       sprintf(str, _("Min time needed: %d"), stats[TIME_NEEDED_STAT]);
148     else// if(i == JUMPS_STAT)
149       sprintf(str, _("Min jumps: %d"), stats[JUMPS_STAT]);
150
151     context.draw_text(white_text, str, Vector(screen->w/2, 430 + i*22), CENTER_ALLIGN, LAYER_GUI);
152     }
153 }
154
155 void
156 Statistics::add_points(int stat, int points)
157 {
158   stats[stat] += points;
159 }
160
161 int
162 Statistics::get_points(int stat)
163 {
164   return stats[stat];
165 }
166
167 void
168 Statistics::set_points(int stat, int points)
169 {
170   stats[stat] = points;
171 }
172
173 void
174 Statistics::reset()
175 {
176   for(int i = 0; i < NUM_STATS; i++)
177     stats[i] = 0;
178 }
179
180 void
181 Statistics::merge(Statistics& stats_)
182 {
183   stats[SCORE_STAT] = std::max(stats[SCORE_STAT], stats_.stats[SCORE_STAT]);
184   if(stats[JUMPS_STAT] != -1)
185     stats[JUMPS_STAT] = my_min(stats[JUMPS_STAT], stats_.stats[JUMPS_STAT]);
186   stats[BADGUYS_SQUISHED_STAT] =
187     std::max(stats[BADGUYS_SQUISHED_STAT], stats_.stats[BADGUYS_SQUISHED_STAT]);
188   stats[SHOTS_STAT] = my_min(stats[SHOTS_STAT], stats_.stats[SHOTS_STAT]);
189   stats[TIME_NEEDED_STAT] =
190     my_min(stats[TIME_NEEDED_STAT], stats_.stats[TIME_NEEDED_STAT]);
191 }
192
193 void
194 Statistics::operator+=(const Statistics& stats_)
195 {
196   for(int i = 0; i < NUM_STATS; i++)
197     stats[i] += stats_.stats[i];
198 }