0f0a5c15411475ef7a233d3d3c1c56f5159912cd
[supertux.git] / src / textscroller.cpp
1 //  $Id$
2 //
3 //  SuperTux
4 //  Copyright (C) 2006 Matthias Braun <matze@braunis.de>
5 //
6 //  This program is free software; you can redistribute it and/or
7 //  modify it under the terms of the GNU General Public License
8 //  as published by the Free Software Foundation; either version 2
9 //  of the License, or (at your option) any later version.
10 //
11 //  This program is distributed in the hope that it will be useful,
12 //  but WITHOUT ANY WARRANTY; without even the implied warranty of
13 //  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 //  GNU General Public License for more details.
15 //
16 //  You should have received a copy of the GNU General Public License
17 //  along with this program; if not, write to the Free Software
18 //  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19 //  02111-1307, USA.
20 #include <config.h>
21
22 #include "textscroller.hpp"
23
24 #include <stdexcept>
25 #include "log.hpp"
26 #include "mainloop.hpp"
27 #include "resources.hpp"
28 #include "video/font.hpp"
29 #include "video/drawing_context.hpp"
30 #include "video/surface.hpp"
31 #include "gui/menu.hpp"
32 #include "lisp/parser.hpp"
33 #include "lisp/lisp.hpp"
34 #include "audio/sound_manager.hpp"
35 #include "main.hpp"
36 #include "fadeout.hpp"
37 #include "control/joystickkeyboardcontroller.hpp"
38
39 static const float DEFAULT_SPEED = 20;
40 static const float LEFT_BORDER = 50;
41 static const float SCROLL = 60;
42 static const float ITEMS_SPACE = 4;
43
44
45
46 TextScroller::TextScroller(const std::string& filename)
47 {
48   defaultspeed = DEFAULT_SPEED;
49   speed = defaultspeed;
50   
51   std::string text;
52   std::string background_file;
53
54   lisp::Parser parser;
55   try {
56     std::auto_ptr<lisp::Lisp> root (parser.parse(filename));
57
58     const lisp::Lisp* text_lisp = root->get_lisp("supertux-text");
59     if(!text_lisp)
60       throw std::runtime_error("File isn't a supertux-text file");
61     
62     if(!text_lisp->get("text", text))
63       throw std::runtime_error("file doesn't contain a text field");
64     if(!text_lisp->get("background", background_file))
65       throw std::runtime_error("file doesn't contain a background file");
66     text_lisp->get("speed", defaultspeed);
67     text_lisp->get("music", music);
68   } catch(std::exception& e) {
69     std::ostringstream msg;
70     msg << "Couldn't load file '" << filename << "': " << e.what() << std::endl;
71     throw std::runtime_error(msg.str());
72   }
73
74   // Split text string lines into a vector
75   lines = InfoBoxLine::split(text, 40);
76
77   // load background image
78   background.reset(new Surface("images/background/" + background_file));
79
80   scroll = 0;
81 }
82
83 TextScroller::~TextScroller()
84 {
85   for(std::vector<InfoBoxLine*>::iterator i = lines.begin(); i != lines.end(); i++) delete *i;
86 }
87
88 void
89 TextScroller::setup()
90 {
91   sound_manager->play_music(music);
92   Menu::set_current(NULL);
93 }
94
95 void
96 TextScroller::update(float elapsed_time)
97 {
98   if(main_controller->hold(Controller::UP)) {
99     speed = -defaultspeed*5;
100   } else if(main_controller->hold(Controller::DOWN)) {
101     speed = defaultspeed*5;
102   } else {
103     speed = defaultspeed;
104   }
105   if(main_controller->pressed(Controller::JUMP)
106       || main_controller->pressed(Controller::ACTION)
107       || main_controller->pressed(Controller::MENU_SELECT))
108     scroll += SCROLL;    
109   if(main_controller->pressed(Controller::PAUSE_MENU)) {
110     main_loop->exit_screen(new FadeOut(0.5));
111   }
112
113   scroll += speed * elapsed_time;
114     
115   if(scroll < 0)
116     scroll = 0;
117 }
118
119 void
120 TextScroller::draw(DrawingContext& context)
121 {
122   context.draw_surface(background.get(), Vector(0,0), 0);
123
124   float y = SCREEN_HEIGHT - scroll;
125   for(size_t i = 0; i < lines.size(); i++) {
126     lines[i]->draw(context, Vector(LEFT_BORDER, y), LAYER_GUI);
127     y += lines[i]->get_height();
128   }
129
130   if(y < 0) {
131     main_loop->exit_screen(new FadeOut(0.5));
132   }
133 }
134
135 InfoBox::InfoBox(const std::string& text)
136   : firstline(0)
137 {
138   // Split text string lines into a vector
139   lines = InfoBoxLine::split(text, 23);
140
141   try
142   {
143     // get the arrow sprites
144     arrow_scrollup   = new Surface("images/engine/menu/scroll-up.png");
145     arrow_scrolldown = new Surface("images/engine/menu/scroll-down.png");
146   }
147   catch (std::exception& e)
148   {
149     log_warning << "Could not load scrolling images: " << e.what() << std::endl;
150     arrow_scrollup = 0;
151     arrow_scrolldown = 0;
152   }
153 }
154
155 InfoBox::~InfoBox()
156 {
157   for(std::vector<InfoBoxLine*>::iterator i = lines.begin(); i != lines.end(); i++) delete *i;
158   delete arrow_scrollup;
159   delete arrow_scrolldown;
160 }
161
162 void
163 InfoBox::draw(DrawingContext& context)
164 {
165   float x1 = 200;
166   float y1 = 100;
167   float width = 400;
168   float height = 200;
169
170   context.draw_filled_rect(Vector(x1, y1), Vector(width, height),
171       Color(0.6f, 0.7f, 0.8f, 0.5f), LAYER_GUI-1);
172
173   float y = y1;
174   for(size_t i = firstline; i < lines.size(); ++i) {
175     if(y >= y1 + height) break;
176
177     lines[i]->draw(context, Vector(x1, y), LAYER_GUI);
178     y += lines[i]->get_height();
179
180     // draw the scrolling arrows
181     if (arrow_scrollup && firstline > 0)
182       context.draw_surface(arrow_scrollup,
183       Vector( x1 + width  - arrow_scrollup->get_width(),  // top-right corner of box
184               y1), LAYER_GUI);
185
186     if (arrow_scrolldown && firstline < lines.size()-1)
187       context.draw_surface(arrow_scrolldown,
188       Vector( x1 + width  - arrow_scrolldown->get_width(),  // bottom-light corner of box
189               y1 + height - arrow_scrolldown->get_height()),
190               LAYER_GUI);
191   }
192 }
193
194 void
195 InfoBox::scrollup()
196 {
197   if(firstline > 0)
198     firstline--;
199 }
200
201 void
202 InfoBox::scrolldown()
203 {
204   if(firstline < lines.size()-1)
205     firstline++;
206 }
207
208 void
209 InfoBox::pageup()
210 {
211 }
212
213 void
214 InfoBox::pagedown()
215 {
216 }
217
218 InfoBoxLine::InfoBoxLine(char format_char, const std::string& text) : lineType(NORMAL), font(white_text), text(text), image(0)
219 {
220   switch(format_char)
221   {
222     case ' ': 
223       lineType = SMALL;
224       font = white_small_text;
225       break;
226     case '\t': 
227       lineType = NORMAL;
228       font = white_text;
229       break;
230     case '-': 
231       lineType = HEADING;
232       font = white_big_text;
233       break;
234     case '*': 
235       lineType = REFERENCE;
236       font = blue_text;
237       break;
238     case '#': 
239       lineType = NORMAL_LEFT;
240       font = white_text;
241       break;
242     case '!': 
243       lineType = IMAGE;
244       image = new Surface(text);
245       break;
246     default:
247       log_warning << "Unknown format_char: '" << format_char << "'" << std::endl;
248       break;
249   }
250 }
251
252 InfoBoxLine::~InfoBoxLine()
253 {
254   delete image;
255 }
256
257 const std::vector<InfoBoxLine*> 
258 InfoBoxLine::split(const std::string& text, int line_length)
259 {
260   std::vector<InfoBoxLine*> lines;
261
262   std::string::size_type i = 0;
263   std::string::size_type l;
264   char format_char = '#';
265   while(i < text.size()) {
266     // take care of empty lines - represent them as blank lines of normal text
267     if (text[i] == '\n') {
268       lines.push_back(new InfoBoxLine('\t', ""));
269       i++;
270       continue;
271     }
272
273     // extract the format_char
274     format_char = text[i];
275     i++;
276     if (i >= text.size()) break;
277
278     // extract one line
279     l = text.find("\n", i);
280     if (l == std::string::npos) l=text.size();
281     std::string s = text.substr(i, l-i);
282     i = l+1;
283
284     // if we are dealing with an image, just store the line
285     if (format_char == '!') {
286       lines.push_back(new InfoBoxLine(format_char, s));
287       continue;
288     } 
289
290     // if we are dealing with text, wrap long lines
291     while ((int)s.length() > line_length) {
292       int split_at = line_length;
293       while ((split_at > 0) && (s[split_at] != ' ')) split_at--;
294       if (split_at == 0) split_at = line_length;
295
296       lines.push_back(new InfoBoxLine(format_char, s.substr(0, split_at)));
297       if (s[split_at] == ' ') split_at++;
298       s = s.substr(split_at);
299     }
300     lines.push_back(new InfoBoxLine(format_char, s));
301
302   }
303
304   return lines;
305 }
306
307 void 
308 InfoBoxLine::draw(DrawingContext& context, const Vector& position, int layer)
309 {
310   switch (lineType) {
311     case IMAGE:
312       context.draw_surface(image, Vector( (SCREEN_WIDTH - image->get_width()) / 2, position.y), layer);
313       break;
314     case NORMAL_LEFT:
315       context.draw_text(font, text, Vector(position.x, position.y), LEFT_ALLIGN, layer);
316       break;
317     default: 
318       context.draw_text(font, text, Vector(SCREEN_WIDTH/2, position.y), CENTER_ALLIGN, layer);
319       break;
320   }
321 }
322
323 float
324 InfoBoxLine::get_height()
325 {
326   switch (lineType) {
327     case IMAGE:
328       return image->get_height() + ITEMS_SPACE;
329     case NORMAL_LEFT:
330       return font->get_height() + ITEMS_SPACE;
331     default: 
332       return font->get_height() + ITEMS_SPACE;
333   }
334 }
335