2013-08-29 11:45:22 +09:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 yvt
|
2013-09-05 00:52:03 +09:00
|
|
|
based on code of pysnip (c) Mathias Kaerlev 2011-2012.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
This file is part of OpenSpades.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
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.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
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.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-29 11:45:22 +09:00
|
|
|
*/
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2016-12-03 18:23:47 +09:00
|
|
|
#include <cmath>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <deque>
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
#include "GameMap.h"
|
|
|
|
#include "GameMapWrapper.h"
|
2017-09-11 20:59:39 +09:00
|
|
|
#include "GameProperties.h"
|
2013-08-18 16:18:06 +09:00
|
|
|
#include "Grenade.h"
|
2016-12-03 18:23:47 +09:00
|
|
|
#include "HitTestDebugger.h"
|
2013-08-18 16:18:06 +09:00
|
|
|
#include "IGameMode.h"
|
|
|
|
#include "IWorldListener.h"
|
2016-12-03 18:23:47 +09:00
|
|
|
#include "Player.h"
|
|
|
|
#include "Weapon.h"
|
|
|
|
#include "World.h"
|
2019-07-17 00:31:00 +09:00
|
|
|
#include <Core/Debug.h>
|
|
|
|
#include <Core/FileManager.h>
|
|
|
|
#include <Core/IStream.h>
|
2014-03-09 01:03:38 +09:00
|
|
|
#include <Core/Settings.h>
|
|
|
|
|
2016-11-19 21:03:51 +09:00
|
|
|
DEFINE_SPADES_SETTING(cg_debugHitTest, "0");
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
namespace spades {
|
|
|
|
namespace client {
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2017-09-11 20:59:39 +09:00
|
|
|
World::World(const std::shared_ptr<GameProperties> &gameProperties)
|
|
|
|
: gameProperties{gameProperties} {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
}
|
2019-07-17 00:31:00 +09:00
|
|
|
World::~World() { SPADES_MARK_FUNCTION(); }
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2015-01-23 18:26:00 +09:00
|
|
|
size_t World::GetNumPlayers() {
|
|
|
|
size_t numPlayers = 0;
|
2019-07-17 00:31:00 +09:00
|
|
|
for (const auto &p : players) {
|
2016-12-03 18:23:47 +09:00
|
|
|
if (p)
|
|
|
|
++numPlayers;
|
2015-01-23 18:26:00 +09:00
|
|
|
}
|
|
|
|
return numPlayers;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
void World::Advance(float dt) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
ApplyBlockActions();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
for (const auto &player : players)
|
|
|
|
if (player)
|
|
|
|
player->Update(dt);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2017-02-20 03:30:10 +09:00
|
|
|
while (!blockRegenerationQueue.empty()) {
|
|
|
|
auto it = blockRegenerationQueue.begin();
|
|
|
|
if (it->first > time) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const IntVector3 &block = it->second;
|
|
|
|
|
|
|
|
if (map && map->IsSolid(block.x, block.y, block.z)) {
|
|
|
|
uint32_t color = map->GetColor(block.x, block.y, block.z);
|
|
|
|
uint32_t health = 100;
|
|
|
|
color = (color & 0xffffff) | (health << 24);
|
|
|
|
map->Set(block.x, block.y, block.z, true, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
blockRegenerationQueueMap.erase(blockRegenerationQueueMap.find(it->second));
|
|
|
|
blockRegenerationQueue.erase(it);
|
|
|
|
}
|
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
std::vector<decltype(grenades.begin())> removedGrenades;
|
|
|
|
for (auto it = grenades.begin(); it != grenades.end(); it++) {
|
|
|
|
Grenade &g = **it;
|
|
|
|
if (g.Update(dt)) {
|
2013-08-18 16:18:06 +09:00
|
|
|
removedGrenades.push_back(it);
|
|
|
|
}
|
|
|
|
}
|
2019-07-17 00:31:00 +09:00
|
|
|
for (auto it : removedGrenades)
|
|
|
|
grenades.erase(it);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
time += dt;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
void World::SetMap(Handle<GameMap> newMap) {
|
2016-12-03 18:23:47 +09:00
|
|
|
if (map == newMap)
|
2013-09-14 12:06:11 +09:00
|
|
|
return;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-09 01:03:38 +09:00
|
|
|
hitTestDebugger.reset();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (map) {
|
2019-07-17 00:31:00 +09:00
|
|
|
mapWrapper.reset();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
map = newMap;
|
2016-12-03 18:23:47 +09:00
|
|
|
if (map) {
|
2019-07-20 16:34:35 +09:00
|
|
|
mapWrapper = stmp::make_unique<GameMapWrapper>(*map);
|
2013-08-18 16:18:06 +09:00
|
|
|
mapWrapper->Rebuild();
|
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
void World::AddGrenade(std::unique_ptr<Grenade> g) {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
grenades.push_back(std::move(g));
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
void World::SetPlayer(int i, std::unique_ptr<Player> p) {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_MARK_FUNCTION();
|
2019-07-17 00:31:00 +09:00
|
|
|
|
|
|
|
players.at(i) = std::move(p);
|
|
|
|
if (listener) {
|
2013-09-21 04:17:34 +09:00
|
|
|
listener->PlayerObjectSet(i);
|
2019-07-17 00:31:00 +09:00
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2019-07-17 00:31:00 +09:00
|
|
|
void World::SetMode(std::unique_ptr<IGameMode> m) { mode = std::move(m); }
|
2017-09-11 20:59:39 +09:00
|
|
|
|
2017-02-20 03:30:10 +09:00
|
|
|
void World::MarkBlockForRegeneration(const IntVector3 &blockLocation) {
|
|
|
|
UnmarkBlockForRegeneration(blockLocation);
|
|
|
|
|
|
|
|
// Regenerate after 10 seconds
|
|
|
|
auto result = blockRegenerationQueue.emplace(time + 10.0f, blockLocation);
|
2017-02-20 04:09:10 +09:00
|
|
|
blockRegenerationQueueMap.emplace(blockLocation, result);
|
2017-02-20 03:30:10 +09:00
|
|
|
}
|
2017-09-11 20:59:39 +09:00
|
|
|
|
2017-02-20 03:30:10 +09:00
|
|
|
void World::UnmarkBlockForRegeneration(const IntVector3 &blockLocation) {
|
|
|
|
auto it = blockRegenerationQueueMap.find(blockLocation);
|
|
|
|
if (it == blockRegenerationQueueMap.end()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
blockRegenerationQueue.erase(it->second);
|
|
|
|
blockRegenerationQueueMap.erase(it);
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
static std::vector<std::vector<CellPos>>
|
|
|
|
ClusterizeBlocks(const std::vector<CellPos> &blocks) {
|
2014-03-30 04:02:34 +09:00
|
|
|
std::unordered_map<CellPos, bool, CellPosHash> blockMap;
|
2016-12-03 18:23:47 +09:00
|
|
|
for (const auto &block : blocks) {
|
2014-03-30 04:02:34 +09:00
|
|
|
blockMap[block] = true;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
std::vector<std::vector<CellPos>> ret;
|
|
|
|
std::deque<decltype(blockMap)::iterator> queue;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
ret.reserve(64);
|
|
|
|
// wish I could `reserve()` queue...
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
std::size_t addedCount = 0;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
for (auto it = blockMap.begin(); it != blockMap.end(); it++) {
|
2014-03-30 04:02:34 +09:00
|
|
|
SPAssert(queue.empty());
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (!it->second)
|
|
|
|
continue;
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(it);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
std::vector<CellPos> outBlocks;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
while (!queue.empty()) {
|
2014-03-30 04:02:34 +09:00
|
|
|
auto blockitem = queue.front();
|
|
|
|
queue.pop_front();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (!blockitem->second)
|
|
|
|
continue;
|
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
auto pos = blockitem->first;
|
|
|
|
outBlocks.emplace_back(pos);
|
|
|
|
blockitem->second = false;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
decltype(blockMap)::iterator nextIt;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
nextIt = blockMap.find(CellPos(pos.x - 1, pos.y, pos.z));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (nextIt != blockMap.end() && nextIt->second) {
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(nextIt);
|
|
|
|
}
|
|
|
|
nextIt = blockMap.find(CellPos(pos.x + 1, pos.y, pos.z));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (nextIt != blockMap.end() && nextIt->second) {
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(nextIt);
|
|
|
|
}
|
|
|
|
nextIt = blockMap.find(CellPos(pos.x, pos.y - 1, pos.z));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (nextIt != blockMap.end() && nextIt->second) {
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(nextIt);
|
|
|
|
}
|
|
|
|
nextIt = blockMap.find(CellPos(pos.x, pos.y + 1, pos.z));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (nextIt != blockMap.end() && nextIt->second) {
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(nextIt);
|
|
|
|
}
|
|
|
|
nextIt = blockMap.find(CellPos(pos.x, pos.y, pos.z - 1));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (nextIt != blockMap.end() && nextIt->second) {
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(nextIt);
|
|
|
|
}
|
|
|
|
nextIt = blockMap.find(CellPos(pos.x, pos.y, pos.z + 1));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (nextIt != blockMap.end() && nextIt->second) {
|
2014-03-30 04:02:34 +09:00
|
|
|
queue.emplace_back(nextIt);
|
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
SPAssert(!outBlocks.empty());
|
|
|
|
addedCount += outBlocks.size();
|
|
|
|
ret.emplace_back(std::move(outBlocks));
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
SPAssert(addedCount == blocks.size());
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2018-12-31 21:34:30 +09:00
|
|
|
return ret;
|
2014-03-30 04:02:34 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
void World::ApplyBlockActions() {
|
2016-12-03 18:23:47 +09:00
|
|
|
for (const auto &creation : createdBlocks) {
|
|
|
|
const auto &pos = creation.first;
|
|
|
|
const auto &color = creation.second;
|
|
|
|
if (map->IsSolid(pos.x, pos.y, pos.z)) {
|
2014-03-30 16:25:17 +09:00
|
|
|
map->Set(pos.x, pos.y, pos.z, true,
|
2016-12-03 18:23:47 +09:00
|
|
|
color.x | (color.y << 8) | (color.z << 16) | (100UL << 24));
|
2014-03-30 16:25:17 +09:00
|
|
|
continue;
|
|
|
|
}
|
2014-03-30 04:02:34 +09:00
|
|
|
mapWrapper->AddBlock(pos.x, pos.y, pos.z,
|
2016-12-03 18:23:47 +09:00
|
|
|
color.x | (color.y << 8) | (color.z << 16) | (100UL << 24));
|
2014-03-30 04:02:34 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
std::vector<CellPos> cells;
|
2016-12-03 18:23:47 +09:00
|
|
|
for (const auto &cell : destroyedBlocks) {
|
|
|
|
if (!map->IsSolid(cell.x, cell.y, cell.z))
|
2014-03-30 04:02:34 +09:00
|
|
|
continue;
|
|
|
|
cells.emplace_back(cell);
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
cells = mapWrapper->RemoveBlocks(cells);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
auto clusters = ClusterizeBlocks(cells);
|
|
|
|
std::vector<IntVector3> cells2;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
for (const auto &cluster : clusters) {
|
2014-03-30 04:02:34 +09:00
|
|
|
cells2.resize(cluster.size());
|
2016-12-03 18:23:47 +09:00
|
|
|
for (std::size_t i = 0; i < cluster.size(); i++) {
|
2014-03-30 04:02:34 +09:00
|
|
|
auto p = cluster[i];
|
|
|
|
cells2[i] = IntVector3(p.x, p.y, p.z);
|
|
|
|
map->Set(p.x, p.y, p.z, false, 0);
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
if (listener)
|
2014-03-30 04:02:34 +09:00
|
|
|
listener->BlocksFell(cells2);
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
createdBlocks.clear();
|
|
|
|
destroyedBlocks.clear();
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
void World::CreateBlock(spades::IntVector3 pos, spades::IntVector3 color) {
|
2014-03-30 04:02:34 +09:00
|
|
|
auto it = destroyedBlocks.find(CellPos(pos.x, pos.y, pos.z));
|
2016-12-03 18:23:47 +09:00
|
|
|
if (it != destroyedBlocks.end())
|
2014-03-30 04:02:34 +09:00
|
|
|
destroyedBlocks.erase(it);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
createdBlocks[CellPos(pos.x, pos.y, pos.z)] = color;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
void World::DestroyBlock(std::vector<spades::IntVector3> &pos) {
|
2013-08-18 16:18:06 +09:00
|
|
|
std::vector<CellPos> cells;
|
2016-12-03 18:23:47 +09:00
|
|
|
bool allowToDestroyLand = pos.size() == 1;
|
|
|
|
for (size_t i = 0; i < pos.size(); i++) {
|
|
|
|
const IntVector3 &p = pos[i];
|
|
|
|
if (p.z >= (allowToDestroyLand ? 63 : 62) || p.z < 0 || p.x < 0 || p.y < 0 ||
|
|
|
|
p.x >= map->Width() || p.y >= map->Height())
|
2013-08-18 16:18:06 +09:00
|
|
|
continue;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-30 04:02:34 +09:00
|
|
|
CellPos cellp(p.x, p.y, p.z);
|
|
|
|
auto it = createdBlocks.find(cellp);
|
2016-12-03 18:23:47 +09:00
|
|
|
if (it != createdBlocks.end())
|
2014-03-30 04:02:34 +09:00
|
|
|
createdBlocks.erase(it);
|
|
|
|
destroyedBlocks.insert(cellp);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
World::PlayerPersistent &World::GetPlayerPersistent(int index) {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPAssert(index >= 0);
|
|
|
|
SPAssert(index < players.size());
|
2019-07-17 00:31:00 +09:00
|
|
|
return playerPersistents.at(index);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
std::vector<IntVector3> World::CubeLine(spades::IntVector3 v1, spades::IntVector3 v2,
|
|
|
|
int maxLength) {
|
2013-08-18 16:18:06 +09:00
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
IntVector3 c = v1;
|
|
|
|
IntVector3 d = v2 - v1;
|
|
|
|
long ixi, iyi, izi, dx, dy, dz, dxi, dyi, dzi;
|
|
|
|
std::vector<IntVector3> ret;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
int VSID = map->Width();
|
|
|
|
SPAssert(VSID == map->Height());
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
int MAXZDIM = map->Depth();
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (d.x < 0)
|
|
|
|
ixi = -1;
|
2013-08-18 16:18:06 +09:00
|
|
|
else
|
2016-12-03 18:23:47 +09:00
|
|
|
ixi = 1;
|
|
|
|
if (d.y < 0)
|
|
|
|
iyi = -1;
|
|
|
|
else
|
|
|
|
iyi = 1;
|
|
|
|
if (d.z < 0)
|
|
|
|
izi = -1;
|
|
|
|
else
|
|
|
|
izi = 1;
|
|
|
|
|
|
|
|
if ((abs(d.x) >= abs(d.y)) && (abs(d.x) >= abs(d.z))) {
|
|
|
|
dxi = 1024;
|
|
|
|
dx = 512;
|
|
|
|
dyi = (long)(!d.y ? 0x3fffffff / VSID : abs(d.x * 1024 / d.y));
|
|
|
|
dy = dyi / 2;
|
|
|
|
dzi = (long)(!d.z ? 0x3fffffff / VSID : abs(d.x * 1024 / d.z));
|
|
|
|
dz = dzi / 2;
|
|
|
|
} else if (abs(d.y) >= abs(d.z)) {
|
|
|
|
dyi = 1024;
|
|
|
|
dy = 512;
|
|
|
|
dxi = (long)(!d.x ? 0x3fffffff / VSID : abs(d.y * 1024 / d.x));
|
|
|
|
dx = dxi / 2;
|
|
|
|
dzi = (long)(!d.z ? 0x3fffffff / VSID : abs(d.y * 1024 / d.z));
|
|
|
|
dz = dzi / 2;
|
|
|
|
} else {
|
|
|
|
dzi = 1024;
|
|
|
|
dz = 512;
|
|
|
|
dxi = (long)(!d.x ? 0x3fffffff / VSID : abs(d.z * 1024 / d.x));
|
|
|
|
dx = dxi / 2;
|
|
|
|
dyi = (long)(!d.y ? 0x3fffffff / VSID : abs(d.z * 1024 / d.y));
|
|
|
|
dy = dyi / 2;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
if (ixi >= 0)
|
|
|
|
dx = dxi - dx;
|
|
|
|
if (iyi >= 0)
|
|
|
|
dy = dyi - dy;
|
|
|
|
if (izi >= 0)
|
|
|
|
dz = dzi - dz;
|
|
|
|
|
|
|
|
while (1) {
|
2013-08-18 16:18:06 +09:00
|
|
|
ret.push_back(c);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (ret.size() == (size_t)maxLength)
|
2013-08-18 16:18:06 +09:00
|
|
|
break;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (c.x == v2.x && c.y == v2.y && c.z == v2.z)
|
2013-08-18 16:18:06 +09:00
|
|
|
break;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if ((dz <= dx) && (dz <= dy)) {
|
2013-08-18 16:18:06 +09:00
|
|
|
c.z += izi;
|
|
|
|
if (c.z < 0 || c.z >= MAXZDIM)
|
|
|
|
break;
|
|
|
|
dz += dzi;
|
2016-12-03 18:23:47 +09:00
|
|
|
} else {
|
|
|
|
if (dx < dy) {
|
2013-08-18 16:18:06 +09:00
|
|
|
c.x += ixi;
|
|
|
|
if ((unsigned long)c.x >= VSID)
|
|
|
|
break;
|
|
|
|
dx += dxi;
|
2016-12-03 18:23:47 +09:00
|
|
|
} else {
|
2013-08-18 16:18:06 +09:00
|
|
|
c.y += iyi;
|
|
|
|
if ((unsigned long)c.y >= VSID)
|
|
|
|
break;
|
|
|
|
dy += dyi;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
return ret;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
World::WeaponRayCastResult World::WeaponRayCast(spades::Vector3 startPos,
|
2019-07-17 00:31:00 +09:00
|
|
|
spades::Vector3 dir,
|
|
|
|
stmp::optional<int> excludePlayerId) {
|
2013-08-18 16:18:06 +09:00
|
|
|
WeaponRayCastResult result;
|
2019-07-17 00:31:00 +09:00
|
|
|
stmp::optional<int> hitPlayer;
|
2013-08-18 16:18:06 +09:00
|
|
|
float hitPlayerDistance = 0.f;
|
2013-10-19 02:00:42 +02:00
|
|
|
hitTag_t hitFlag = hit_None;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
for (int i = 0; i < (int)players.size(); i++) {
|
2019-07-17 00:31:00 +09:00
|
|
|
const auto &p = players[i];
|
|
|
|
if (!p || (excludePlayerId && *excludePlayerId == i))
|
2013-08-18 16:18:06 +09:00
|
|
|
continue;
|
2016-12-03 18:23:47 +09:00
|
|
|
if (p->GetTeamId() >= 2 || !p->IsAlive())
|
2013-08-18 16:18:06 +09:00
|
|
|
continue;
|
2016-12-03 18:23:47 +09:00
|
|
|
if (!p->RayCastApprox(startPos, dir))
|
2013-08-18 16:18:06 +09:00
|
|
|
continue;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
Player::HitBoxes hb = p->GetHitBoxes();
|
|
|
|
Vector3 hitPos;
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (hb.head.RayCast(startPos, dir, &hitPos)) {
|
2013-08-18 16:18:06 +09:00
|
|
|
float dist = (hitPos - startPos).GetLength();
|
2019-07-17 00:31:00 +09:00
|
|
|
if (!hitPlayer || dist < hitPlayerDistance) {
|
|
|
|
if (hitPlayer != i) {
|
|
|
|
hitPlayer = i;
|
2013-10-19 02:00:42 +02:00
|
|
|
hitFlag = hit_None;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
hitPlayerDistance = dist;
|
2013-10-19 02:00:42 +02:00
|
|
|
hitFlag |= hit_Head;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
if (hb.torso.RayCast(startPos, dir, &hitPos)) {
|
2013-08-18 16:18:06 +09:00
|
|
|
float dist = (hitPos - startPos).GetLength();
|
2019-07-17 00:31:00 +09:00
|
|
|
if (!hitPlayer || dist < hitPlayerDistance) {
|
|
|
|
if (hitPlayer != i) {
|
|
|
|
hitPlayer = i;
|
2013-10-19 02:00:42 +02:00
|
|
|
hitFlag = hit_None;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
hitPlayerDistance = dist;
|
2013-10-19 02:00:42 +02:00
|
|
|
hitFlag |= hit_Torso;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
for (int j = 0; j < 3; j++) {
|
|
|
|
if (hb.limbs[j].RayCast(startPos, dir, &hitPos)) {
|
2013-08-18 16:18:06 +09:00
|
|
|
float dist = (hitPos - startPos).GetLength();
|
2019-07-17 00:31:00 +09:00
|
|
|
if (!hitPlayer || dist < hitPlayerDistance) {
|
|
|
|
if (hitPlayer != i) {
|
|
|
|
hitPlayer = i;
|
2013-10-19 02:00:42 +02:00
|
|
|
hitFlag = hit_None;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
hitPlayerDistance = dist;
|
2016-12-03 18:23:47 +09:00
|
|
|
if (j == 2) {
|
2013-10-19 02:00:42 +02:00
|
|
|
hitFlag |= hit_Arms;
|
|
|
|
} else {
|
|
|
|
hitFlag |= hit_Legs;
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
// map raycast
|
|
|
|
GameMap::RayCastResult res2;
|
|
|
|
res2 = map->CastRay2(startPos, dir, 256);
|
2016-12-03 18:23:47 +09:00
|
|
|
|
|
|
|
if (res2.hit &&
|
2019-07-17 00:31:00 +09:00
|
|
|
(!hitPlayer || (res2.hitPos - startPos).GetLength() < hitPlayerDistance)) {
|
2013-08-18 16:18:06 +09:00
|
|
|
result.hit = true;
|
|
|
|
result.startSolid = res2.startSolid;
|
2013-10-19 02:00:42 +02:00
|
|
|
result.hitFlag = hit_None;
|
2013-08-18 16:18:06 +09:00
|
|
|
result.blockPos = res2.hitBlock;
|
|
|
|
result.hitPos = res2.hitPos;
|
2016-12-03 18:23:47 +09:00
|
|
|
} else if (hitPlayer) {
|
2013-08-18 16:18:06 +09:00
|
|
|
result.hit = true;
|
|
|
|
result.startSolid = false; // FIXME: startSolid for player
|
2019-07-17 00:31:00 +09:00
|
|
|
result.playerId = hitPlayer;
|
2013-08-18 16:18:06 +09:00
|
|
|
result.hitPos = startPos + dir * hitPlayerDistance;
|
2013-10-19 02:00:42 +02:00
|
|
|
result.hitFlag = hitFlag;
|
2016-12-03 18:23:47 +09:00
|
|
|
} else {
|
2013-08-18 16:18:06 +09:00
|
|
|
result.hit = false;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
return result;
|
|
|
|
}
|
2016-12-03 18:23:47 +09:00
|
|
|
|
2014-03-09 01:03:38 +09:00
|
|
|
HitTestDebugger *World::GetHitTestDebugger() {
|
2016-12-03 18:23:47 +09:00
|
|
|
if (cg_debugHitTest) {
|
|
|
|
if (hitTestDebugger == nullptr) {
|
2019-07-20 16:34:35 +09:00
|
|
|
hitTestDebugger = stmp::make_unique<HitTestDebugger>(this);
|
2014-03-09 01:03:38 +09:00
|
|
|
}
|
|
|
|
return hitTestDebugger.get();
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
2019-07-17 00:31:00 +09:00
|
|
|
} // namespace client
|
|
|
|
} // namespace spades
|