2013-09-05 00:52:03 +09:00

378 lines
10 KiB
C++

/*
Copyright (c) 2013 yvt
based on code of pysnip (c) Mathias Kaerlev 2011-2012.
This file is part of OpenSpades.
OpenSpades is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
OpenSpades is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
*/
#include "Weapon.h"
#include "World.h"
#include "IWorldListener.h"
#include "../Core/Exception.h"
#include "../Core/Debug.h"
#include "../Core/Settings.h"
SPADES_SETTING(cg_protocolVersion, "3");
namespace spades {
namespace client {
Weapon::Weapon(World *w, Player *p):
world(w),
owner(p),
time(0),
shooting(false),
reloading(false),
nextShotTime(0.f),
reloadEndTime(-100.f),
reloadStartTime(-101.f),
lastDryFire(false)
{
SPADES_MARK_FUNCTION();
}
Weapon::~Weapon(){
SPADES_MARK_FUNCTION();
}
void Weapon::Restock() {
stock = GetMaxStock();
}
void Weapon::Reset() {
SPADES_MARK_FUNCTION();
shooting = false;
reloading = false;
ammo = GetClipSize();
stock = GetMaxStock();
time = 0.f;
nextShotTime = 0.f;
}
void Weapon::SetShooting(bool b){
shooting = b;
}
bool Weapon::IsReadyToShoot() {
return ammo > 0 && time >= nextShotTime &&
(!reloading || IsReloadSlow());
}
float Weapon::GetReloadProgress() {
return (time - reloadStartTime) / (reloadEndTime - reloadStartTime);
}
float Weapon::TimeToNextFire() {
return nextShotTime - time;
}
bool Weapon::FrameNext(float dt){
SPADES_MARK_FUNCTION();
bool fired = false;
bool dryFire = false;
if(shooting && (!reloading || IsReloadSlow())){
// abort slow reload
reloading = false;
if(time >= nextShotTime && ammo > 0) {
fired = true;
ammo--;
if(world->GetListener())
world->GetListener()->PlayerFiredWeapon(owner);
nextShotTime = time + GetDelay();
}else if(time >= nextShotTime){
dryFire = true;
}
}
if(reloading){
if(time >= reloadEndTime) {
// reload done
reloading = false;
if(IsReloadSlow()){
// for local player, server sends
// new ammo/stock value
if(ammo < GetClipSize() && stock > 0 &&
owner != owner->GetWorld()->GetLocalPlayer()) {
ammo++;
stock--;
}
slowReloadLeftCount--;
if(slowReloadLeftCount > 0)
Reload(false);
else
if(world->GetListener())
world->GetListener()->PlayerReloadedWeapon(owner);
}else{
// for local player, server sends
// new ammo/stock value
if(owner != owner->GetWorld()->GetLocalPlayer()){
int newStock;
newStock = std::max(0, stock - GetClipSize() + ammo);
ammo += stock - newStock;
stock = newStock;
}
if(world->GetListener())
world->GetListener()->PlayerReloadedWeapon(owner);
}
}
}
time += dt;
if(dryFire && !lastDryFire){
if(world->GetListener())
world->GetListener()->PlayerDryFiredWeapon(owner);
}
lastDryFire = dryFire;
return fired;
}
void Weapon::ReloadDone(int ammo, int stock) {
SPADES_MARK_FUNCTION_DEBUG();
this->ammo = ammo;
this->stock = stock;
}
void Weapon::AbortReload() {
SPADES_MARK_FUNCTION_DEBUG();
reloading = false;
}
void Weapon::Reload(bool manual) {
SPADES_MARK_FUNCTION();
if(reloading)
return;
if(ammo >= GetClipSize())
return;
if(stock == 0)
return;
if(IsReloadSlow() && ammo > 0 && shooting)
return;
if(manual)
slowReloadLeftCount = stock - std::max(0, stock - GetClipSize() + ammo);
reloading = true;
shooting = false;
reloadStartTime = time;
reloadEndTime = time + GetReloadTime();
if(world->GetListener())
world->GetListener()->PlayerReloadingWeapon(owner);
}
void Weapon::ForceReloadDone() {
int newStock;
newStock = std::max(0, stock - GetClipSize() + ammo);
ammo += stock - newStock;
stock = newStock;
}
class RifleWeapon3: public Weapon {
public:
RifleWeapon3(World*w,Player*p):Weapon(w,p){}
virtual std::string GetName() { return "Rifle"; }
virtual float GetDelay() { return 0.5f; }
virtual int GetClipSize() { return 10; }
virtual int GetMaxStock() { return 50; }
virtual float GetReloadTime() { return 2.5f; }
virtual bool IsReloadSlow() { return false; }
virtual WeaponType GetWeaponType() { return RIFLE_WEAPON; }
virtual int GetDamage(HitType type, float distance) {
switch(type){
case HitTypeTorso: return 49;
case HitTypeHead: return 100;
case HitTypeArms: return 33;
case HitTypeLegs: return 33;
case HitTypeBlock: return 50;
}
}
virtual Vector3 GetRecoil () {
return MakeVector3(0.0001f, 0.05f, 0.f);
}
virtual float GetSpread() { return 0.006f; }
virtual int GetPelletSize() { return 1; }
};
class SMGWeapon3: public Weapon {
public:
SMGWeapon3(World*w,Player*p):Weapon(w,p){}
virtual std::string GetName() { return "SMG"; }
virtual float GetDelay() { return 0.1f; }
virtual int GetClipSize() { return 30; }
virtual int GetMaxStock() { return 120; }
virtual float GetReloadTime() { return 2.5f; }
virtual bool IsReloadSlow() { return false; }
virtual WeaponType GetWeaponType() { return SMG_WEAPON; }
virtual int GetDamage(HitType type, float distance) {
switch(type){
case HitTypeTorso: return 29;
case HitTypeHead: return 75;
case HitTypeArms: return 18;
case HitTypeLegs: return 18;
case HitTypeBlock: return 25;
}
}
virtual Vector3 GetRecoil () {
return MakeVector3(0.00005f, 0.0125f, 0.f);
}
virtual float GetSpread() { return 0.012f; }
virtual int GetPelletSize() { return 1; }
};
class ShotgunWeapon3: public Weapon {
public:
ShotgunWeapon3(World*w,Player*p):Weapon(w,p){}
virtual std::string GetName() { return "Shotgun"; }
virtual float GetDelay() { return 1.f; }
virtual int GetClipSize() { return 6; }
virtual int GetMaxStock() { return 48; }
virtual float GetReloadTime() { return 0.5f; }
virtual bool IsReloadSlow() { return true; }
virtual WeaponType GetWeaponType() { return SHOTGUN_WEAPON; }
virtual int GetDamage(HitType type, float distance) {
switch(type){
case HitTypeTorso: return 27;
case HitTypeHead: return 37;
case HitTypeArms: return 16;
case HitTypeLegs: return 16;
case HitTypeBlock:
// Actually, you cast a hit per pallet. This value is a guess, by the way. --GM
return 34;
}
}
virtual Vector3 GetRecoil () {
return MakeVector3(0.0002f, 0.1f, 0.f);
}
virtual float GetSpread() { return 0.024f; }
virtual int GetPelletSize() { return 8; }
};
class RifleWeapon4: public Weapon {
public:
RifleWeapon4(World*w,Player*p):Weapon(w,p){}
virtual std::string GetName() { return "Rifle"; }
virtual float GetDelay() { return 0.6f; }
virtual int GetClipSize() { return 8; }
virtual int GetMaxStock() { return 48; }
virtual float GetReloadTime() { return 2.5f; }
virtual bool IsReloadSlow() { return false; }
virtual WeaponType GetWeaponType() { return RIFLE_WEAPON; }
virtual int GetDamage(HitType type, float distance) {
switch(type){
// These are the 0.75 damage values.
// To be honest, we don't need this information, as the server decides the damage.
// EXCEPT for blocks, that is.
// --GM
case HitTypeTorso: return 49;
case HitTypeHead: return 100;
case HitTypeArms: return 33;
case HitTypeLegs: return 33;
case HitTypeBlock: return 50;
}
}
virtual Vector3 GetRecoil () {
return MakeVector3(0.0001f, 0.075f, 0.f);
}
virtual float GetSpread() { return 0.004f; }
virtual int GetPelletSize() { return 1; }
};
class SMGWeapon4: public Weapon {
public:
SMGWeapon4(World*w,Player*p):Weapon(w,p){}
virtual std::string GetName() { return "SMG"; }
virtual float GetDelay() { return 0.1f; }
virtual int GetClipSize() { return 30; }
virtual int GetMaxStock() { return 150; }
virtual float GetReloadTime() { return 2.5f; }
virtual bool IsReloadSlow() { return false; }
virtual WeaponType GetWeaponType() { return SMG_WEAPON; }
virtual int GetDamage(HitType type, float distance) {
switch(type){
case HitTypeTorso: return 29;
case HitTypeHead: return 75;
case HitTypeArms: return 18;
case HitTypeLegs: return 18;
case HitTypeBlock: return 34;
}
}
virtual Vector3 GetRecoil () {
return MakeVector3(0.00005f, 0.0125f, 0.f);
}
virtual float GetSpread() { return 0.012f; }
virtual int GetPelletSize() { return 1; }
};
class ShotgunWeapon4: public Weapon {
public:
ShotgunWeapon4(World*w,Player*p):Weapon(w,p){}
virtual std::string GetName() { return "Shotgun"; }
virtual float GetDelay() { return 0.8f; }
virtual int GetClipSize() { return 8; }
virtual int GetMaxStock() { return 48; }
virtual float GetReloadTime() { return 0.4f; }
virtual bool IsReloadSlow() { return true; }
virtual WeaponType GetWeaponType() { return SHOTGUN_WEAPON; }
virtual int GetDamage(HitType type, float distance) {
switch(type){
case HitTypeTorso: return 27;
case HitTypeHead: return 37;
case HitTypeArms: return 16;
case HitTypeLegs: return 16;
case HitTypeBlock:
return 34;
}
}
virtual Vector3 GetRecoil () {
return MakeVector3(0.0002f, 0.075f, 0.f);
}
virtual float GetSpread() { return 0.036f; }
virtual int GetPelletSize() { return 8; }
};
Weapon *Weapon::CreateWeapon(WeaponType type, Player *p) {
SPADES_MARK_FUNCTION();
if((int)cg_protocolVersion == 4) {
switch(type) {
case RIFLE_WEAPON:
return new RifleWeapon4(p->GetWorld(), p);
case SMG_WEAPON:
return new SMGWeapon4(p->GetWorld(), p);
case SHOTGUN_WEAPON:
return new ShotgunWeapon4(p->GetWorld(), p);
default:
SPInvalidEnum("type", type);
}
} else {
switch(type) {
case RIFLE_WEAPON:
return new RifleWeapon3(p->GetWorld(), p);
case SMG_WEAPON:
return new SMGWeapon3(p->GetWorld(), p);
case SHOTGUN_WEAPON:
return new ShotgunWeapon3(p->GetWorld(), p);
default:
SPInvalidEnum("type", type);
}
}
}
}
}