WillOWisp badguys can now be trapped inside empty Lanterns.
[supertux.git] / src / object / rock.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 "rock.hpp"
23 #include "lisp/writer.hpp"
24 #include "object_factory.hpp"
25 #include "audio/sound_manager.hpp"
26 #include "tile.hpp"
27
28 namespace {
29   const std::string ROCK_SOUND = "sounds/brick.wav"; //TODO use own sound.
30 }
31
32 Rock::Rock(const lisp::Lisp& reader)
33   : MovingSprite(reader, "images/objects/rock/rock.sprite")
34 {
35   sound_manager->preload(ROCK_SOUND);
36   on_ground = false;
37   grabbed = false;
38   set_group(COLGROUP_MOVING_STATIC);
39 }
40
41 Rock::Rock(const lisp::Lisp& reader, std::string spritename)
42   : MovingSprite(reader, spritename)
43 {
44   sound_manager->preload(ROCK_SOUND);
45   on_ground = false;
46   grabbed = false;
47   set_group(COLGROUP_MOVING_STATIC);
48 }
49
50 void
51 Rock::write(lisp::Writer& writer)
52 {
53   writer.start_list("rock");
54
55   writer.write_float("x", bbox.p1.x);
56   writer.write_float("y", bbox.p1.y);
57
58   writer.end_list("rock");
59 }
60
61 void
62 Rock::update(float elapsed_time)
63 {
64   if( grabbed )
65     return;
66
67   if (on_ground) physic.set_velocity_x(0);
68
69   movement = physic.get_movement(elapsed_time);
70 }
71
72 void
73 Rock::collision_solid(const CollisionHit& hit)
74 {
75   if(grabbed) {
76     return;
77   }
78   if(hit.top || hit.bottom)
79     physic.set_velocity_y(0);
80   if(hit.left || hit.right)
81     physic.set_velocity_x(0);
82   if(hit.crush)
83     physic.set_velocity(0, 0);
84
85   if(hit.bottom  && !on_ground) {
86     sound_manager->play(ROCK_SOUND, get_pos());
87     on_ground = true;
88   }
89 }
90
91 HitResponse
92 Rock::collision(GameObject& other, const CollisionHit& hit)
93 {
94   if(grabbed) {
95     return PASSTHROUGH;
96   }
97   if(!on_ground) {
98     if(hit.bottom && physic.get_velocity_y() > 200) {
99       MovingObject* moving_object = dynamic_cast<MovingObject*> (&other);
100       if(moving_object) {
101         //Getting a rock on the head hurts. A lot.
102         moving_object->collision_tile(Tile::HURTS);
103       }
104     }
105     return FORCE_MOVE;
106   }
107
108   return FORCE_MOVE;
109 }
110
111 void
112 Rock::grab(MovingObject& , const Vector& pos, Direction)
113 {
114   movement = pos - get_pos();
115   last_movement = movement;
116   set_group(COLGROUP_TOUCHABLE);
117   on_ground = true;
118   grabbed = true;
119 }
120
121 void
122 Rock::ungrab(MovingObject& , Direction dir)
123 {
124   set_group(COLGROUP_MOVING_STATIC);
125   on_ground = false;
126   if(dir == UP) {
127     physic.set_velocity(0, -500);
128   } else if (last_movement.norm() > 1) {
129     physic.set_velocity((dir == RIGHT) ? 200 : -200, -200);
130   } else {
131     physic.set_velocity(0, 0);
132   }
133   grabbed = false;
134 }
135
136 IMPLEMENT_FACTORY(Rock, "rock");