* B140: Hack alarm: "fixed" the black-fading screen as requested by jonaskoelker.
[supertux.git] / src / scripting / functions.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  02111-1307, USA.
19
20 #include <config.h>
21
22 #include <memory>
23 #include <stdio.h>
24 #include <string>
25 #include <squirrel.h>
26 #include <sqstdio.h>
27 #include "textscroller.hpp"
28 #include "functions.hpp"
29 #include "game_session.hpp"
30 #include "tinygettext/tinygettext.hpp"
31 #include "physfs/physfs_stream.hpp"
32 #include "resources.hpp"
33 #include "gettext.hpp"
34 #include "log.hpp"
35 #include "mainloop.hpp"
36 #include "worldmap/worldmap.hpp"
37 #include "world.hpp"
38 #include "sector.hpp"
39 #include "gameconfig.hpp"
40 #include "object/player.hpp"
41 #include "object/tilemap.hpp"
42 #include "main.hpp"
43 #include "fadeout.hpp"
44 #include "shrinkfade.hpp"
45 #include "object/camera.hpp"
46 #include "flip_level_transformer.hpp"
47 #include "audio/sound_manager.hpp"
48 #include "random_generator.hpp"
49
50 #include "squirrel_error.hpp"
51 #include "squirrel_util.hpp"
52 #include "time_scheduler.hpp"
53
54 namespace Scripting
55 {
56
57 SQInteger display(HSQUIRRELVM vm)
58 {
59   Console::output << squirrel2string(vm, -1) << std::endl;
60   return 0;
61 }
62
63 void print_stacktrace(HSQUIRRELVM vm)
64 {
65   print_squirrel_stack(vm);
66 }
67
68 SQInteger get_current_thread(HSQUIRRELVM vm)
69 {
70   sq_pushobject(vm, vm_to_object(vm));
71   return 1;
72 }
73
74 void wait(HSQUIRRELVM vm, float seconds)
75 {
76   TimeScheduler::instance->schedule_thread(vm, game_time + seconds);
77 }
78
79 void wait_for_screenswitch(HSQUIRRELVM vm)
80 {
81   main_loop->waiting_threads.add(vm);
82 }
83
84 void exit_screen()
85 {
86   main_loop->exit_screen();
87 }
88
89 void fadeout_screen(float seconds)
90 {
91   main_loop->set_screen_fade(new FadeOut(seconds));
92 }
93
94 void shrink_screen(float dest_x, float dest_y, float seconds)
95 {
96   main_loop->set_screen_fade(new ShrinkFade(Vector(dest_x, dest_y), seconds));
97 }
98
99 void abort_screenfade()
100 {
101   main_loop->set_screen_fade(NULL);
102 }
103
104 std::string translate(const std::string& text)
105 {
106   return dictionary_manager.get_dictionary().translate(text);
107 }
108
109 void display_text_file(const std::string& filename)
110 {
111   main_loop->push_screen(new TextScroller(filename));
112 }
113
114 void load_worldmap(const std::string& filename)
115 {
116   using namespace WorldMapNS;
117
118   main_loop->push_screen(new WorldMap(filename));
119 }
120
121 void load_level(const std::string& filename)
122 {
123   main_loop->push_screen(new GameSession(filename));
124 }
125
126 static SQInteger squirrel_read_char(SQUserPointer file)
127 {
128   std::istream* in = reinterpret_cast<std::istream*> (file);
129   char c = in->get();
130   if(in->eof())
131     return 0;
132
133   return c;
134 }
135
136 void import(HSQUIRRELVM vm, const std::string& filename)
137 {
138   IFileStream in(filename);
139
140   if(SQ_FAILED(sq_compile(vm, squirrel_read_char, &in,
141           filename.c_str(), SQTrue)))
142     throw SquirrelError(vm, "Couldn't parse script");
143
144   sq_pushroottable(vm);
145   if(SQ_FAILED(sq_call(vm, 1, SQFalse, SQTrue))) {
146     sq_pop(vm, 1);
147     throw SquirrelError(vm, "Couldn't execute script");
148   }
149   sq_pop(vm, 1);
150 }
151
152 void debug_collrects(bool enable)
153 {
154   Sector::show_collrects = enable;
155 }
156
157 void debug_show_fps(bool enable)
158 {
159   config->show_fps = enable;
160 }
161
162 void debug_draw_solids_only(bool enable)
163 {
164   Sector::draw_solids_only = enable;
165 }
166
167 void save_state()
168 {
169   using namespace WorldMapNS;
170
171   if(World::current() == NULL || WorldMap::current() == NULL)
172     throw std::runtime_error("Can't save state without active World");
173
174   WorldMap::current()->save_state();
175   World::current()->save_state();
176 }
177
178 void update_worldmap()
179 {
180   using namespace WorldMapNS;
181
182   if(WorldMap::current() == NULL)
183     throw std::runtime_error("Can't update Worldmap: none active");
184
185   WorldMap::current()->load_state();
186 }
187
188 // not added to header, function to only be used by others
189 // in this file
190 bool validate_sector_player()
191 {
192   if (Sector::current() == 0)
193   {
194     log_info << "No current sector." << std::endl;
195         return false;
196   }
197
198   if (Sector::current()->player == 0)
199   {
200     log_info << "No player." << std::endl;
201         return false;
202   }
203   return true;
204 }
205
206 void play_music(const std::string& filename)
207 {
208   sound_manager->play_music(filename);
209 }
210
211 void play_sound(const std::string& filename)
212 {
213   sound_manager->play(filename);
214 }
215
216 void grease()
217 {
218   if (!validate_sector_player()) return;
219   ::Player* tux = Sector::current()->player; // Scripting::Player != ::Player
220   tux->physic.set_velocity_x(tux->physic.get_velocity_x()*3);
221 }
222
223 void invincible()
224 {
225   if (!validate_sector_player()) return;
226   ::Player* tux = Sector::current()->player;
227   tux->invincible_timer.start(10000);
228 }
229
230 void ghost()
231 {
232   if (!validate_sector_player()) return;
233   ::Player* tux = Sector::current()->player;
234   tux->set_ghost_mode(true);
235 }
236
237 void mortal()
238 {
239   if (!validate_sector_player()) return;
240   ::Player* tux = Sector::current()->player;
241   tux->invincible_timer.stop();
242   tux->set_ghost_mode(false);
243 }
244
245 void restart()
246 {
247   if (GameSession::current() == 0)
248   {
249     log_info << "No game session" << std::endl;
250     return;
251   }
252   GameSession::current()->restart_level();
253 }
254
255 void whereami()
256 {
257   if (!validate_sector_player()) return;
258   ::Player* tux = Sector::current()->player;
259   log_info << "You are at x " << tux->get_pos().x << ", y " << tux->get_pos().y << std::endl;
260 }
261
262 void gotoend()
263 {
264   if (!validate_sector_player()) return;
265   ::Player* tux = Sector::current()->player;
266   tux->move(Vector(
267           (Sector::current()->get_width()) - (SCREEN_WIDTH*2), 0));
268   Sector::current()->camera->reset(
269         Vector(tux->get_pos().x, tux->get_pos().y));
270 }
271
272 void camera()
273 {
274   if (!validate_sector_player()) return;
275   log_info << "Camera is at " << Sector::current()->camera->get_translation().x << "," << Sector::current()->camera->get_translation().y << std::endl;
276 }
277
278 void quit()
279 {
280   main_loop->quit();
281 }
282
283 int rand()
284 {
285   return systemRandom.rand();
286 }
287
288 }