ddnet/src/game/g_game.cpp

405 lines
9.1 KiB
C++
Raw Normal View History

2007-11-25 19:42:40 +00:00
/* copyright (c) 2007 magnus auvinen, see licence.txt for more info */
2007-12-15 10:24:49 +00:00
#include "g_game.h"
2007-09-22 18:55:00 +00:00
// TODO: OPT: rewrite this smarter!
void move_point(vec2 *inout_pos, vec2 *inout_vel, float elasticity, int *bounces)
{
if(bounces)
*bounces = 0;
vec2 pos = *inout_pos;
vec2 vel = *inout_vel;
if(col_check_point(pos + vel))
{
int affected = 0;
if(col_check_point(pos.x + vel.x, pos.y))
{
inout_vel->x *= -elasticity;
if(bounces)
(*bounces)++;
affected++;
}
if(col_check_point(pos.x, pos.y + vel.y))
{
inout_vel->y *= -elasticity;
if(bounces)
(*bounces)++;
affected++;
}
if(affected == 0)
{
inout_vel->x *= -elasticity;
inout_vel->y *= -elasticity;
}
}
else
{
*inout_pos = pos + vel;
}
}
2007-12-12 19:52:57 +00:00
bool test_box(vec2 pos, vec2 size)
2007-09-25 20:04:54 +00:00
{
size *= 0.5f;
if(col_check_point(pos.x-size.x, pos.y-size.y))
return true;
if(col_check_point(pos.x+size.x, pos.y-size.y))
return true;
if(col_check_point(pos.x-size.x, pos.y+size.y))
return true;
if(col_check_point(pos.x+size.x, pos.y+size.y))
return true;
return false;
}
2007-09-22 18:55:00 +00:00
void move_box(vec2 *inout_pos, vec2 *inout_vel, vec2 size, float elasticity)
{
// do the move
vec2 pos = *inout_pos;
vec2 vel = *inout_vel;
float distance = length(vel);
int max = (int)distance;
if(distance > 0.00001f)
{
2007-09-25 20:04:54 +00:00
//vec2 old_pos = pos;
float fraction = 1.0f/(float)(max+1);
2007-09-22 18:55:00 +00:00
for(int i = 0; i <= max; i++)
{
2007-09-25 20:04:54 +00:00
//float amount = i/(float)max;
//if(max == 0)
//amount = 0;
2007-09-22 18:55:00 +00:00
2007-09-25 20:04:54 +00:00
vec2 new_pos = pos + vel*fraction; // TODO: this row is not nice
2007-12-12 20:05:18 +00:00
if(test_box(vec2(new_pos.x, new_pos.y), size))
2007-09-22 18:55:00 +00:00
{
2007-12-12 20:05:18 +00:00
int hits = 0;
if(test_box(vec2(pos.x, new_pos.y), size))
2007-09-22 18:55:00 +00:00
{
2007-09-25 20:04:54 +00:00
new_pos.y = pos.y;
2007-09-25 21:53:14 +00:00
vel.y *= -elasticity;
2007-12-12 20:05:18 +00:00
hits++;
}
if(test_box(vec2(new_pos.x, pos.y), size))
{
new_pos.x = pos.x;
vel.x *= -elasticity;
hits++;
2007-09-25 20:04:54 +00:00
}
2007-12-12 20:05:18 +00:00
// neither of the tests got a collision.
// this is a real _corner case_!
if(hits == 0)
2007-09-25 20:04:54 +00:00
{
2007-12-12 20:05:18 +00:00
new_pos.y = pos.y;
vel.y *= -elasticity;
2007-09-25 20:04:54 +00:00
new_pos.x = pos.x;
2007-09-25 21:53:14 +00:00
vel.x *= -elasticity;
2007-09-22 18:55:00 +00:00
}
}
2007-09-25 20:04:54 +00:00
pos = new_pos;
2007-09-22 18:55:00 +00:00
}
}
*inout_pos = pos;
*inout_vel = vel;
}
void player_core::tick()
{
float phys_size = 28.0f;
2007-12-09 12:40:34 +00:00
triggered_events = 0;
2007-09-22 18:55:00 +00:00
bool grounded = false;
if(col_is_solid((int)(pos.x+phys_size/2), (int)(pos.y+phys_size/2+5)))
2007-09-22 18:55:00 +00:00
grounded = true;
if(col_is_solid((int)(pos.x-phys_size/2), (int)(pos.y+phys_size/2+5)))
2007-09-22 18:55:00 +00:00
grounded = true;
vec2 direction = normalize(vec2(input.target_x, input.target_y));
vel.y += gravity;
float max_speed = grounded ? ground_control_speed : air_control_speed;
float accel = grounded ? ground_control_accel : air_control_accel;
float friction = grounded ? ground_friction : air_friction;
// handle movement
if(input.left)
vel.x = saturated_add(-max_speed, max_speed, vel.x, -accel);
if(input.right)
vel.x = saturated_add(-max_speed, max_speed, vel.x, accel);
if(!input.left && !input.right)
vel.x *= friction;
// handle jumping
2007-12-09 09:48:53 +00:00
// 1 bit = to keep track if a jump has been made on this input
// 2 bit = to keep track if a air-jump has been made
if(grounded)
jumped &= ~2;
2007-09-22 18:55:00 +00:00
if(input.jump)
{
2007-12-09 09:48:53 +00:00
if(!(jumped&1))
2007-09-22 18:55:00 +00:00
{
2007-12-09 09:48:53 +00:00
if(grounded)
{
2007-12-09 12:40:34 +00:00
triggered_events |= COREEVENT_GROUND_JUMP;
2007-12-09 09:48:53 +00:00
vel.y = -ground_jump_speed;
jumped |= 1;
}
else if(!(jumped&2))
{
2007-12-09 12:40:34 +00:00
triggered_events |= COREEVENT_AIR_JUMP;
2007-12-09 09:48:53 +00:00
vel.y = -ground_air_speed;
jumped |= 3;
}
2007-09-22 18:55:00 +00:00
}
}
else
2007-12-09 09:48:53 +00:00
jumped &= ~1;
2007-09-22 18:55:00 +00:00
// do hook
if(input.hook)
{
if(hook_state == HOOK_IDLE)
{
hook_state = HOOK_FLYING;
hook_pos = pos;
hook_dir = direction;
hooked_player = -1;
hook_tick = 0;
2007-12-09 12:40:34 +00:00
triggered_events |= COREEVENT_HOOK_LAUNCH;
2007-09-22 18:55:00 +00:00
}
else if(hook_state == HOOK_FLYING)
{
vec2 new_pos = hook_pos+hook_dir*hook_fire_speed;
// Check against other players first
for(int i = 0; i < MAX_CLIENTS; i++)
{
player_core *p = world->players[i];
if(!p || p == this)
continue;
//if(p != this && !p->dead && distance(p->pos, new_pos) < p->phys_size)
if(distance(p->pos, new_pos) < phys_size)
{
2007-12-09 12:40:34 +00:00
triggered_events |= COREEVENT_HOOK_ATTACH_PLAYER;
2007-09-22 18:55:00 +00:00
hook_state = HOOK_GRABBED;
hooked_player = i;
break;
}
}
/*
for(entity *ent = world.first_entity; ent; ent = ent->next_entity)
{
if(ent && ent->objtype == OBJTYPE_PLAYER)
{
player *p = (player*)ent;
if(p != this && !p->dead && distance(p->pos, new_pos) < p->phys_size)
{
hook_state = HOOK_GRABBED;
hooked_player = p;
break;
}
}
}*/
if(hook_state == HOOK_FLYING)
{
// check against ground
if(col_intersect_line(hook_pos, new_pos, &new_pos))
{
2007-12-09 12:40:34 +00:00
triggered_events |= COREEVENT_HOOK_ATTACH_GROUND;
2007-09-22 18:55:00 +00:00
hook_state = HOOK_GRABBED;
hook_pos = new_pos;
}
else if(distance(pos, new_pos) > hook_length)
{
2007-12-09 12:40:34 +00:00
triggered_events |= COREEVENT_HOOK_RETRACT;
2007-09-22 18:55:00 +00:00
hook_state = HOOK_RETRACTED;
}
else
hook_pos = new_pos;
}
if(hook_state == HOOK_GRABBED)
{
//create_sound(pos, SOUND_HOOK_ATTACH);
//hook_tick = server_tick();
}
}
}
else
{
//release_hooked();
hooked_player = -1;
hook_state = HOOK_IDLE;
hook_pos = pos;
}
2007-09-22 18:55:00 +00:00
if(hook_state == HOOK_GRABBED)
{
if(hooked_player != -1)
{
player_core *p = world->players[hooked_player];
if(p)
hook_pos = p->pos;
else
{
// release hook
hooked_player = -1;
hook_state = HOOK_RETRACTED;
hook_pos = pos;
}
2007-09-22 18:55:00 +00:00
// keep players hooked for a max of 1.5sec
//if(server_tick() > hook_tick+(server_tickspeed()*3)/2)
//release_hooked();
}
// don't do this hook rutine when we are hook to a player
if(hooked_player == -1 && distance(hook_pos, pos) > 46.0f)
2007-09-22 18:55:00 +00:00
{
vec2 hookvel = normalize(hook_pos-pos)*hook_drag_accel;
// the hook as more power to drag you up then down.
// this makes it easier to get on top of an platform
if(hookvel.y > 0)
hookvel.y *= 0.3f;
// the hook will boost it's power if the player wants to move
// in that direction. otherwise it will dampen everything abit
if((hookvel.x < 0 && input.left) || (hookvel.x > 0 && input.right))
hookvel.x *= 0.95f;
else
hookvel.x *= 0.75f;
vec2 new_vel = vel+hookvel;
// check if we are under the legal limit for the hook
if(length(new_vel) < hook_drag_speed || length(new_vel) < length(vel))
vel = new_vel; // no problem. apply
2007-09-22 18:55:00 +00:00
}
// release hook
hook_tick++;
if(hooked_player != -1 && hook_tick > SERVER_TICK_SPEED*2)
{
hooked_player = -1;
hook_state = HOOK_RETRACTED;
hook_pos = pos;
}
2007-09-22 18:55:00 +00:00
}
if(true)
{
for(int i = 0; i < MAX_CLIENTS; i++)
{
player_core *p = world->players[i];
if(!p)
continue;
//player *p = (player*)ent;
if(p == this) // || !(p->flags&FLAG_ALIVE)
continue; // make sure that we don't nudge our self
// handle player <-> player collision
float d = distance(pos, p->pos);
vec2 dir = normalize(pos - p->pos);
if(d < phys_size*1.25f && d > 1.0f)
2007-09-22 18:55:00 +00:00
{
float a = phys_size*1.25f - d;
// make sure that we don't add excess force by checking the
// direction against the current velocity
vec2 veldir = normalize(vel);
float v = 1-(dot(veldir, dir)+1)/2;
vel = vel + dir*a*v;
2007-09-22 18:55:00 +00:00
}
// handle hook influence
if(hooked_player == i)
{
if(d > phys_size*1.50f) // TODO: fix tweakable variable
{
float accel = hook_drag_accel * (d/hook_length);
// add force to the hooked player
p->vel.x = saturated_add(-hook_drag_speed, hook_drag_speed, p->vel.x, accel*dir.x*1.5f);
p->vel.y = saturated_add(-hook_drag_speed, hook_drag_speed, p->vel.y, accel*dir.y*1.5f);
// add a little bit force to the guy who has the grip
vel.x = saturated_add(-hook_drag_speed, hook_drag_speed, vel.x, -accel*dir.x*0.25f);
vel.y = saturated_add(-hook_drag_speed, hook_drag_speed, vel.y, -accel*dir.y*0.25f);
2007-09-22 18:55:00 +00:00
}
}
}
}
}
void player_core::move()
{
move_box(&pos, &vel, vec2(28.0f, 28.0f), 0);
}
void player_core::write(obj_player_core *obj_core)
{
obj_core->x = (int)pos.x;
obj_core->y = (int)pos.y;
obj_core->vx = (int)(vel.x*256.0f);
obj_core->vy = (int)(vel.y*256.0f);
obj_core->hook_state = hook_state;
obj_core->hook_tick = hook_tick;
2007-09-22 18:55:00 +00:00
obj_core->hook_x = (int)hook_pos.x;
obj_core->hook_y = (int)hook_pos.y;
obj_core->hook_dx = (int)(hook_dir.x*256.0f);
obj_core->hook_dy = (int)(hook_dir.y*256.0f);
obj_core->hooked_player = hooked_player;
2007-12-09 09:48:53 +00:00
obj_core->jumped = jumped;
2007-09-22 18:55:00 +00:00
float a = 0;
if(input.target_x == 0)
a = atan((float)input.target_y);
else
a = atan((float)input.target_y/(float)input.target_x);
if(input.target_x < 0)
a = a+pi;
obj_core->angle = (int)(a*256.0f);
}
void player_core::read(const obj_player_core *obj_core)
{
pos.x = obj_core->x;
pos.y = obj_core->y;
vel.x = obj_core->vx/256.0f;
vel.y = obj_core->vy/256.0f;
hook_state = obj_core->hook_state;
hook_tick = obj_core->hook_tick;
2007-09-22 18:55:00 +00:00
hook_pos.x = obj_core->hook_x;
hook_pos.y = obj_core->hook_y;
hook_dir.x = obj_core->hook_dx/256.0f;
hook_dir.y = obj_core->hook_dy/256.0f;
hooked_player = obj_core->hooked_player;
2007-12-09 09:48:53 +00:00
jumped = obj_core->jumped;
2007-09-22 18:55:00 +00:00
}
void player_core::quantize()
{
obj_player_core c;
write(&c);
read(&c);
}