1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
/*
game/ship.cc
This file is part of the Osirion project and is distributed under
the terms and conditions of the GNU General Public License version 2
*/
// project headers
#include "game/game.h"
#include "game/ship.h"
#include "math/mathlib.h"
// C++ headers
#include <iostream>
using math::degrees360f;
using math::degrees180f;
namespace game {
Ship::Ship(core::Player *owner, ShipModel *shipmodel) :
core::EntityControlable(owner, ship_enttype)
{
entity_modelname = "ships/" + shipmodel->modelname();
entity_name = shipmodel->name() + ": <" + owner->name() + ">";
ship_shipmodel = shipmodel;
entity_moduletypeid = ship_enttype;
current_target_direction = 0.0f;
current_target_pitch = 0.0f;;
current_target_roll = 0.0f;;
}
Ship::~Ship()
{
if (entity_owner)
entity_owner->remove_asset(this);
}
void Ship::frame(float seconds)
{
const float direction_change_speed = 2;
// update thrust
math::clamp(target_thrust, 0.0f, 1.0f);
entity_thrust = target_thrust;
// update pitch
math::clamp(target_pitch, -1.0f, 1.0f);
if (current_target_pitch < target_pitch) {
current_target_pitch += direction_change_speed * seconds;
if (current_target_pitch > target_pitch)
current_target_pitch = target_pitch;
} else if (current_target_pitch > target_pitch) {
current_target_pitch -= direction_change_speed * seconds;
if (current_target_pitch < target_pitch)
current_target_pitch = target_pitch;
}
if (fabs(seconds*current_target_pitch) > 0.0f) {
math::clamp(current_target_pitch, -1.0f, 1.0f);
float pitch_offset = seconds * current_target_pitch;
entity_axis.change_pitch(360.0f * ship_shipmodel->turnspeed() * pitch_offset);
}
// update direction
math::clamp(target_direction, -1.0f, 1.0f);
if (current_target_direction < target_direction) {
current_target_direction += direction_change_speed * seconds;
if (current_target_direction > target_direction) {
current_target_direction = target_direction;
}
} else if (current_target_direction > target_direction) {
current_target_direction -= direction_change_speed * seconds;
if (current_target_direction < target_direction) {
current_target_direction = target_direction;
}
}
if (fabs(seconds*current_target_direction) > 0.0f ) {
math::clamp(current_target_direction, -1.0f, 1.0f);
float direction_offset = seconds * current_target_direction;
entity_axis.change_direction(360.0f * ship_shipmodel->turnspeed() * direction_offset);
}
// update roll
math::clamp(target_roll, -1.0f, 1.0f);
if (current_target_roll < target_roll) {
current_target_roll += direction_change_speed * seconds;
if (current_target_roll > target_roll)
current_target_roll = target_roll;
} else if (current_target_roll > target_roll) {
current_target_roll -= direction_change_speed * seconds;
if (current_target_roll < target_roll)
current_target_roll = target_roll;
}
if (fabs(current_target_roll) > 0.0f) {
math::clamp(current_target_roll, -1.0f, 1.0f);
float roll_offset = seconds * current_target_roll;
entity_axis.change_roll(360.0f * ship_shipmodel->turnspeed() * roll_offset);
}
// update speed
if (entity_speed < entity_thrust * ship_shipmodel->maxspeed()) {
entity_speed += ship_shipmodel->acceleration() * seconds;
if (entity_speed > entity_thrust * ship_shipmodel->maxspeed()) {
entity_speed = entity_thrust * ship_shipmodel->maxspeed();
}
} else if(entity_speed > entity_thrust * ship_shipmodel->maxspeed()) {
entity_speed -= ship_shipmodel->acceleration() * seconds;
if (entity_speed < 0.0f) entity_speed = 0.0f;
}
if (entity_speed != 0.0f) {
entity_location += entity_axis.forward() * entity_speed * seconds;
entity_dirty = true;
} else if ((current_target_pitch != 0.0f) || (current_target_direction != 0.0f) || (current_target_roll != 0.0f)) {
entity_dirty = true;
}
}
} // namespace game
|