Alayan 553595fa63
Prevent geometry level from removing objects
The latest version was bugged as it removed all standard objects, but removing only objects marked by track-makers removed almost nothing of note, and bugs would sometimes remove normal objects. Fix #5052. Fix #5066. This makes #5063 moot.
2024-05-06 22:59:58 +02:00

3063 lines
107 KiB
C++

// SuperTuxKart - a fun racing game with go-kart
//
// Copyright (C) 2004-2015 Steve Baker <sjbaker1@airmail.net>
// Copyright (C) 2009-2015 Joerg Henrichs, Steve Baker
//
// This program 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.
//
// This program 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 this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#include "tracks/track.hpp"
#include "addons/addon.hpp"
#include "audio/music_manager.hpp"
#include "challenges/challenge_status.hpp"
#include "challenges/unlock_manager.hpp"
#include "config/player_manager.hpp"
#include "config/stk_config.hpp"
#include "config/user_config.hpp"
#include "graphics/camera_end.hpp"
#include "graphics/CBatchingMesh.hpp"
#include "graphics/central_settings.hpp"
#include "graphics/cpu_particle_manager.hpp"
#include "graphics/irr_driver.hpp"
#include "graphics/lod_node.hpp"
#include "graphics/material.hpp"
#include "graphics/material_manager.hpp"
#include "graphics/mesh_tools.hpp"
#include "graphics/moving_texture.hpp"
#include "graphics/particle_emitter.hpp"
#include "graphics/particle_kind.hpp"
#include "graphics/particle_kind_manager.hpp"
#include "graphics/render_target.hpp"
#include "graphics/shader_based_renderer.hpp"
#include "graphics/shader_files_manager.hpp"
#include "graphics/stk_tex_manager.hpp"
#include "graphics/sp/sp_base.hpp"
#include "graphics/sp/sp_mesh.hpp"
#include "graphics/sp/sp_mesh_buffer.hpp"
#include "graphics/sp/sp_mesh_node.hpp"
#include "graphics/sp/sp_shader_manager.hpp"
#include "graphics/sp/sp_texture_manager.hpp"
#include "io/file_manager.hpp"
#include "io/xml_node.hpp"
#include "items/item.hpp"
#include "items/item_manager.hpp"
#include "items/network_item_manager.hpp"
#include "items/powerup_manager.hpp"
#include "karts/abstract_kart.hpp"
#include "karts/kart_properties.hpp"
#include "main_loop.hpp"
#include "modes/linear_world.hpp"
#include "modes/easter_egg_hunt.hpp"
#include "network/network_config.hpp"
#include "network/protocols/game_protocol.hpp"
#include "network/protocols/server_lobby.hpp"
#include "physics/physical_object.hpp"
#include "physics/physics.hpp"
#include "physics/triangle_mesh.hpp"
#include "race/race_manager.hpp"
#include "scriptengine/script_engine.hpp"
#include "tracks/arena_graph.hpp"
#include "tracks/bezier_curve.hpp"
#include "tracks/check_manager.hpp"
#include "tracks/check_structure.hpp"
#include "tracks/drive_graph.hpp"
#include "tracks/drive_node.hpp"
#include "tracks/model_definition_loader.hpp"
#include "tracks/track_manager.hpp"
#include "tracks/track_object_manager.hpp"
#include "utils/constants.hpp"
#include "utils/log.hpp"
#include "mini_glm.hpp"
#include "utils/string_utils.hpp"
#include "utils/translation.hpp"
#include <IBillboardTextSceneNode.h>
#include <IFileSystem.h>
#include <ILightSceneNode.h>
#include <IMeshCache.h>
#include <IMeshManipulator.h>
#include <IMeshSceneNode.h>
#include <ISceneManager.h>
#include <IVideoDriver.h>
#include <SMeshBuffer.h>
#include <algorithm>
#include <iostream>
#include <stdexcept>
#include <sstream>
#include <wchar.h>
#ifndef SERVER_ONLY
#include <ge_main.hpp>
#include <ge_texture.hpp>
#endif
using namespace irr;
const float Track::NOHIT = -99999.9f;
bool Track::m_dont_load_navmesh = false;
std::atomic<Track*> Track::m_current_track[PT_COUNT];
// ----------------------------------------------------------------------------
Track::Track(const std::string &filename)
{
#ifdef DEBUG
m_magic_number = 0x17AC3802;
#endif
m_minimap_invert_x_z = false;
m_materials_loaded = false;
m_filename = filename;
m_root =
StringUtils::getPath(StringUtils::removeExtension(m_filename));
m_ident = StringUtils::getBasename(m_root);
// If this is an addon track, add "addon_" to the identifier - just in
// case that an addon track has the same directory name (and therefore
// identifier) as an included track.
if(Addon::isAddon(filename))
{
m_ident = Addon::createAddonId(m_ident);
m_is_addon = true;
}
else
m_is_addon = false;
// The directory should always have a '/' at the end, but getBasename
// above returns "" if a "/" is at the end, so we add the "/" here.
m_root += "/";
m_designer = "";
m_screenshot = "";
m_version = 0;
m_track_mesh = NULL;
m_height_map_mesh = NULL;
m_gfx_effect_mesh = NULL;
m_internal = false;
m_enable_auto_rescue = true; // Below set to false in arenas
m_enable_push_back = true;
m_reverse_available = false;
m_is_arena = false;
m_is_ctf = false;
m_max_arena_players = 0;
m_has_easter_eggs = false;
m_has_navmesh = false;
m_is_soccer = false;
m_is_cutscene = false;
m_camera_far = 1000.0f;
m_bloom = true;
m_is_day = true;
m_bloom_threshold = 0.75f;
m_color_inlevel = core::vector3df(0.0,1.0, 255.0);
m_color_outlevel = core::vector2df(0.0, 255.0);
m_godrays = false;
m_displacement_speed = 1.0f;
m_physical_object_uid = 0;
m_shadows = true;
m_sky_particles = NULL;
m_sky_dx = 0.05f;
m_sky_dy = 0.0f;
m_godrays_opacity = 1.0f;
m_godrays_color = video::SColor(255, 255, 255, 255);
m_weather_lightning = false;
m_weather_sound = "";
m_cache_track = UserConfigParams::m_cache_overworld &&
m_ident=="overworld";
m_render_target = NULL;
m_check_manager = NULL;
m_minimap_x_scale = 1.0f;
m_minimap_y_scale = 1.0f;
m_force_disable_fog = false;
m_startup_run = false;
m_music_idx = 0;
m_red_flag = m_blue_flag =
btTransform(btQuaternion(0.0f, 0.0f, 0.0f, 1.0f));
m_default_number_of_laps = 3;
m_all_nodes.clear();
m_static_physics_only_nodes.clear();
m_all_cached_meshes.clear();
loadTrackInfo();
} // Track
//-----------------------------------------------------------------------------
/** Destructor, removes quad data structures etc. */
Track::~Track()
{
// Note that the music information in m_music is globally managed
// by the music_manager, and is freed there. So no need to free it
// here (esp. since various track might share the same music).
#ifdef DEBUG
assert(m_magic_number == 0x17AC3802);
m_magic_number = 0xDEADBEEF;
#endif
} // ~Track
//-----------------------------------------------------------------------------
/** A < comparison of tracks. This is used to sort the tracks when displaying
* them in the gui.
*/
bool Track::operator<(const Track &other) const
{
PlayerProfile *p = PlayerManager::getCurrentPlayer();
bool this_is_locked = p->isLocked(getIdent());
bool other_is_locked = p->isLocked(other.getIdent());
if(this_is_locked == other_is_locked)
{
return getSortName() < other.getSortName();
}
else
return other_is_locked;
} // operator<
//-----------------------------------------------------------------------------
/** Returns the name of the track, which is e.g. displayed on the screen. */
core::stringw Track::getName() const
{
core::stringw translated = _(m_name.c_str());
int index = translated.find("|");
if(index>-1)
{
translated = translated.subString(0, index);
}
return translated;
} // getName
//-----------------------------------------------------------------------------
/** Returns the name of the track used to sort the tracks alphabetically.
* This can be used to e.g. sort 'The Island' as 'Island,The'; or
* to replace certain language-specific characters (e.g. German 'ae' with 'a')
* The sort name can be specified by setting the name of a track to:
* "normal name|sort name"
*/
core::stringw Track::getSortName() const
{
core::stringw translated = translations->w_gettext(m_name.c_str());
translated.make_lower();
int index = translated.find("|");
if(index>-1)
{
translated = translated.subString(index+1, translated.size());
}
return translated;
} // getSortName
//-----------------------------------------------------------------------------
/** Returns true if this track belongs to the specified track group.
* \param group_name Group name to test for.
*/
bool Track::isInGroup(const std::string &group_name)
{
return std::find(m_groups.begin(), m_groups.end(), group_name)
!= m_groups.end();
} // isInGroup
//-----------------------------------------------------------------------------
/** Returns number of completed challenges */
unsigned int Track::getNumOfCompletedChallenges()
{
unsigned int unlocked_challenges = 0;
PlayerProfile *player = PlayerManager::getCurrentPlayer();
for (unsigned int i=0; i<m_challenges.size(); i++)
{
if (m_challenges[i].m_challenge_id == "tutorial")
{
unlocked_challenges++;
continue;
}
if (player->getChallengeStatus(m_challenges[i].m_challenge_id)
->isSolvedAtAnyDifficulty())
{
unlocked_challenges++;
}
}
return unlocked_challenges;
} // getNumOfCompletedChallenges
//-----------------------------------------------------------------------------
/** Removes all cached data structures. This is called before the resolution
* is changed.
*/
void Track::removeCachedData()
{
m_materials_loaded = false;
} // cleanCachedData
//-----------------------------------------------------------------------------
/** Prepates the track for a new race. This function must be called after all
* karts are created, since the check objects allocate data structures
* depending on the number of karts.
*/
void Track::reset()
{
m_ambient_color = m_default_ambient_color;
m_check_manager->reset(*this);
m_item_manager->reset();
m_track_object_manager->reset();
m_startup_run = false;
} // reset
//-----------------------------------------------------------------------------
/** Removes the physical body from the world.
* Called at the end of a race.
*/
void Track::cleanup()
{
irr_driver->resetSceneComplexity();
m_physical_object_uid = 0;
#ifdef USE_RESIZE_CACHE
if (!UserConfigParams::m_high_definition_textures)
{
file_manager->popTextureSearchPath();
}
#endif
file_manager->popTextureSearchPath();
file_manager->popModelSearchPath();
Graph::destroy();
m_item_manager = nullptr;
#ifndef SERVER_ONLY
if (CVS->isGLSL())
{
if (!GUIEngine::isNoGraphics())
{
CPUParticleManager::getInstance()->cleanMaterialMap();
}
SP::resetEmptyFogColor();
}
ParticleKindManager::get()->cleanUpTrackSpecificGfx();
#endif
for (unsigned int i = 0; i < m_animated_textures.size(); i++)
{
delete m_animated_textures[i];
}
m_animated_textures.clear();
for (unsigned int i = 0; i < m_all_nodes.size(); i++)
{
irr_driver->removeNode(m_all_nodes[i]);
}
m_all_nodes.clear();
for (unsigned int i = 0; i < m_static_physics_only_nodes.size(); i++)
{
m_static_physics_only_nodes[i]->remove();
}
m_static_physics_only_nodes.clear();
delete m_check_manager;
m_check_manager = NULL;
delete m_track_object_manager;
m_track_object_manager = NULL;
for (unsigned int i = 0; i < m_object_physics_only_nodes.size(); i++)
{
m_object_physics_only_nodes[i]->drop();
}
m_object_physics_only_nodes.clear();
#ifndef SERVER_ONLY
irr_driver->removeNode(m_sun);
if (CVS->isGLSL())
m_sun->drop();
#endif
delete m_track_mesh;
m_track_mesh = NULL;
delete m_height_map_mesh;
m_height_map_mesh = NULL;
delete m_gfx_effect_mesh;
m_gfx_effect_mesh = NULL;
#ifndef SERVER_ONLY
if (CVS->isGLSL())
irr_driver->cleanSunInterposer();
#endif
// The m_all_cached_mesh contains each mesh loaded from a file, which
// means that the mesh is stored in irrlichts mesh cache. To clean
// everything loaded by this track, we drop the ref count for each mesh
// here, till the ref count is 1, which means the mesh is only contained
// in the mesh cache, and can therefore be removed. Meshes load more
// than once are in m_all_cached_mesh more than once (which is easier
// than storing the mesh only once, but then having to test for each
// mesh if it is already contained in the list or not).
for (unsigned int i = 0; i < m_all_cached_meshes.size(); i++)
{
irr_driver->dropAllTextures(m_all_cached_meshes[i]);
// If a mesh is not in Irrlicht's texture cache, its refcount is
// 1 (since its scene node was removed, so the only other reference
// is in m_all_cached_meshes). In this case we only drop it once
// and don't try to remove it from the cache.
if (m_all_cached_meshes[i]->getReferenceCount() == 1)
{
m_all_cached_meshes[i]->drop();
continue;
}
m_all_cached_meshes[i]->drop();
if (m_all_cached_meshes[i]->getReferenceCount() == 1)
irr_driver->removeMeshFromCache(m_all_cached_meshes[i]);
}
m_all_cached_meshes.clear();
// Now free meshes that are not associated to any scene node.
for (unsigned int i = 0; i < m_detached_cached_meshes.size(); i++)
{
irr_driver->dropAllTextures(m_detached_cached_meshes[i]);
irr_driver->removeMeshFromCache(m_detached_cached_meshes[i]);
}
m_detached_cached_meshes.clear();
for(unsigned int i=0; i<m_sky_textures.size(); i++)
{
video::ITexture* tex = (video::ITexture*)m_sky_textures[i];
tex->drop();
if (tex->getReferenceCount() == 1)
irr_driver->removeTexture(tex);
}
m_sky_textures.clear();
if(m_cache_track)
material_manager->makeMaterialsPermanent();
else
{
// remove temporary materials loaded by the material manager
material_manager->popTempMaterial();
}
#ifndef SERVER_ONLY
irr_driver->clearGlowingNodes();
irr_driver->clearLights();
irr_driver->clearForcedBloom();
irr_driver->clearBackgroundNodes();
if (CVS->isGLSL())
{
SP::SPShaderManager::get()->removeUnusedShaders();
ShaderFilesManager::getInstance()->removeUnusedShaderFiles();
SP::SPTextureManager::get()->removeUnusedTextures();
}
#endif
if(UserConfigParams::logMemory())
{
Log::debug("track",
"[memory] After cleaning '%s': mesh cache %d texture cache %d\n",
getIdent().c_str(),
irr_driver->getSceneManager()->getMeshCache()->getMeshCount(),
irr_driver->getVideoDriver()->getTextureCount());
#ifdef DEBUG
scene::IMeshCache *cache = irr_driver->getSceneManager()->getMeshCache();
for(unsigned int i=0; i<cache->getMeshCount(); i++)
{
const io::SNamedPath &name = cache->getMeshName(i);
std::vector<std::string>::iterator p;
p = std::find(m_old_mesh_buffers.begin(), m_old_mesh_buffers.end(),
name.getInternalName().c_str());
if(p!=m_old_mesh_buffers.end())
m_old_mesh_buffers.erase(p);
else
{
Log::debug("track", "[memory] Leaked mesh buffer '%s'.\n",
name.getInternalName().c_str());
} // if name not found
} // for i < cache size
video::IVideoDriver *vd = irr_driver->getVideoDriver();
for(unsigned int i=0; i<vd->getTextureCount(); i++)
{
video::ITexture *t = vd->getTextureByIndex(i);
std::vector<video::ITexture*>::iterator p;
p = std::find(m_old_textures.begin(), m_old_textures.end(),
t);
if(p!=m_old_textures.end())
{
m_old_textures.erase(p);
}
else
{
Log::debug("track", "[memory] Leaked texture '%s'.\n",
t->getName().getInternalName().c_str());
}
}
#endif
} // if verbose
#ifdef __DEBUG_DUMP_MESH_CACHE_AFTER_CLEANUP__
scene::IMeshCache* meshCache = irr_driver->getSceneManager()->getMeshCache();
int count = meshCache->getMeshCount();
for (int i = 0; i < count; i++)
{
scene::IAnimatedMesh* mesh = meshCache->getMeshByIndex(i);
io::SNamedPath path = meshCache->getMeshName(mesh);
Log::info("CACHE", "[%i] %s", i, path.getPath().c_str());
}
#endif
m_meta_library.clear();
Scripting::ScriptEngine::getInstance()->cleanupCache();
m_current_track[PT_MAIN] = NULL;
} // cleanup
//-----------------------------------------------------------------------------
void Track::loadTrackInfo()
{
// Default values
m_use_fog = false;
m_fog_max = 1.0f;
m_fog_start = 0.0f;
m_fog_end = 1000.0f;
m_fog_height_start = 0.0f;
m_fog_height_end = 100.0f;
m_gravity = 9.80665f;
m_friction = stk_config->m_default_track_friction;
m_smooth_normals = false;
m_godrays = false;
m_godrays_opacity = 1.0f;
m_godrays_color = video::SColor(255, 255, 255, 255);
/* ARGB */
m_fog_color = video::SColor(255, 77, 179, 230);
m_default_ambient_color = video::SColor(255, 120, 120, 120);
m_sun_specular_color = video::SColor(255, 255, 255, 255);
m_sun_diffuse_color = video::SColor(255, 255, 255, 255);
m_sun_position = core::vector3df(0, 10, 10);
irr_driver->setSSAORadius(1.);
irr_driver->setSSAOK(1.5);
irr_driver->setSSAOSigma(1.);
XMLNode *root = file_manager->createXMLTree(m_filename);
if(!root || root->getName()!="track")
{
delete root;
std::ostringstream o;
o<<"Can't load track '"<<m_filename<<"', no track element.";
throw std::runtime_error(o.str());
}
root->get("name", &m_name);
std::string designer;
root->get("designer", &designer);
m_designer = StringUtils::xmlDecode(designer);
root->get("version", &m_version);
std::vector<std::string> filenames;
root->get("music", &filenames);
root->get("screenshot", &m_screenshot);
root->get("gravity", &m_gravity);
root->get("friction", &m_friction);
root->get("soccer", &m_is_soccer);
root->get("arena", &m_is_arena);
root->get("ctf", &m_is_ctf);
root->get("max-arena-players", &m_max_arena_players);
root->get("cutscene", &m_is_cutscene);
root->get("groups", &m_groups);
root->get("internal", &m_internal);
root->get("reverse", &m_reverse_available);
root->get("default-number-of-laps",&m_default_number_of_laps);
root->get("push-back", &m_enable_push_back);
root->get("bloom", &m_bloom);
root->get("bloom-threshold", &m_bloom_threshold);
root->get("shadows", &m_shadows);
root->get("is-during-day", &m_is_day);
root->get("displacement-speed", &m_displacement_speed);
root->get("color-level-in", &m_color_inlevel);
root->get("color-level-out", &m_color_outlevel);
getMusicInformation(filenames, m_music);
if (m_default_number_of_laps <= 0)
m_default_number_of_laps = 3;
m_actual_number_of_laps = m_default_number_of_laps;
// Make the default for auto-rescue in battle mode and soccer mode to be false
if(m_is_arena || m_is_soccer)
m_enable_auto_rescue = false;
root->get("auto-rescue", &m_enable_auto_rescue);
root->get("smooth-normals", &m_smooth_normals);
// Reverse is meaningless in arena
if(m_is_arena || m_is_soccer)
m_reverse_available = false;
for(unsigned int i=0; i<root->getNumNodes(); i++)
{
const XMLNode *mode=root->getNode(i);
if(mode->getName()!="mode") continue;
TrackMode tm;
mode->get("name", &tm.m_name );
mode->get("quads", &tm.m_quad_name );
mode->get("graph", &tm.m_graph_name);
mode->get("scene", &tm.m_scene );
m_all_modes.push_back(tm);
}
// If no mode is specified, add a default mode.
if(m_all_modes.size()==0)
{
TrackMode tm;
m_all_modes.push_back(tm);
}
if(m_groups.size()==0) m_groups.push_back(DEFAULT_GROUP_NAME);
const XMLNode *xml_node = root->getNode("curves");
if(xml_node) loadCurves(*xml_node);
// Set the correct paths
if (m_screenshot.length() > 0)
m_screenshot = m_root+m_screenshot;
delete root;
std::string dir = StringUtils::getPath(m_filename);
std::string easter_name = dir + "/easter_eggs.xml";
XMLNode *easter = file_manager->createXMLTree(easter_name);
if(easter)
{
for(unsigned int i=0; i<easter->getNumNodes(); i++)
{
const XMLNode *eggs = easter->getNode(i);
if(eggs->getNumNodes() > 0)
{
m_has_easter_eggs = true;
break;
}
}
delete easter;
}
if(file_manager->fileExists(m_root+"navmesh.xml") && !m_dont_load_navmesh)
m_has_navmesh = true;
else if ( (m_is_arena || m_is_soccer) && !m_dont_load_navmesh)
{
Log::warn("Track", "NavMesh is not found for arena %s, "
"disable AI for it.\n", m_name.c_str());
}
if (m_is_soccer)
{
// Currently only max eight players in soccer mode
m_max_arena_players = 8;
}
// Max 10 players supported in arena
if (m_max_arena_players > 10)
m_max_arena_players = 10;
} // loadTrackInfo
//-----------------------------------------------------------------------------
/** Loads all curves from the XML node.
*/
void Track::loadCurves(const XMLNode &node)
{
for(unsigned int i=0; i<node.getNumNodes(); i++)
{
const XMLNode *curve = node.getNode(i);
m_all_curves.push_back(new BezierCurve(*curve));
} // for i<node.getNumNodes
} // loadCurves
//-----------------------------------------------------------------------------
/** Loads all music information for the specified files (which is taken from
* the track.xml file).
* \param filenames List of filenames to load.
* \param music On return contains the music information object for the
* specified files.
*/
void Track::getMusicInformation(std::vector<std::string>& filenames,
std::vector<MusicInformation*>& music )
{
for(int i=0; i<(int)filenames.size(); i++)
{
std::string full_path = m_root+filenames[i];
MusicInformation* mi = music_manager->getMusicInformation(full_path);
if(!mi)
{
try
{
std::string shared_name = file_manager->searchMusic(filenames[i]);
if(shared_name!="")
mi = music_manager->getMusicInformation(shared_name);
}
catch (...)
{
mi = NULL;
}
}
if(mi)
m_music.push_back(mi);
else
Log::warn("track",
"Music information file '%s' not found for track '%s' - ignored.\n",
filenames[i].c_str(), m_name.c_str());
} // for i in filenames
if (m_music.empty() && !isInternal() && !m_is_cutscene)
{
m_music.push_back(stk_config->m_default_music);
Log::warn("track",
"Music information for track '%s' replaced by default music.\n",
m_name.c_str());
}
if (!m_music.empty())
m_music_idx = rand() % m_music.size();
} // getMusicInformation
//-----------------------------------------------------------------------------
/** Select and set the music for this track (doesn't actually start it yet).
*/
void Track::startMusic() const
{
// In case that the music wasn't found (a warning was already printed)
if(m_music.size()>0)
music_manager->startMusic(m_music[m_music_idx], false);
else
music_manager->clearCurrentMusic();
} // startMusic
//-----------------------------------------------------------------------------
/** Loads the quad graph for arena, i.e. the definition of all quads, and the
* way they are connected to each other. Input file name is hardcoded for now
*/
void Track::loadArenaGraph(const XMLNode &node)
{
// Determine if rotate minimap is needed for soccer mode (for blue team)
// Only need to test local player
if (RaceManager::get()->isSoccerMode())
{
const unsigned pk = RaceManager::get()->getNumPlayers();
for (unsigned i = 0; i < pk; i++)
{
if (!RaceManager::get()->getKartInfo(i).isNetworkPlayer() &&
RaceManager::get()->getKartInfo(i).getKartTeam() ==
KART_TEAM_BLUE)
{
m_minimap_invert_x_z = true;
break;
}
}
}
ArenaGraph* graph = new ArenaGraph(m_root+"navmesh.xml", &node);
Graph::setGraph(graph);
if(Graph::get()->getNumNodes()==0)
{
Log::warn("track", "No graph nodes defined for track '%s'\n",
m_filename.c_str());
}
else
{
loadMinimap();
}
} // loadArenaGraph
//-----------------------------------------------------------------------------
btQuaternion Track::getArenaStartRotation(const Vec3& xyz, float heading)
{
btQuaternion def_pos(Vec3(0, 1, 0), heading * DEGREE_TO_RAD);
if (!ArenaGraph::get())
return def_pos;
// Set the correct axis based on normal of the starting position
int node = Graph::UNKNOWN_SECTOR;
Graph::get()->findRoadSector(xyz, &node);
if (node == Graph::UNKNOWN_SECTOR)
{
Log::warn("track", "Starting position is not on ArenaGraph");
return def_pos;
}
const Vec3& normal = Graph::get()->getQuad(node)->getNormal();
btQuaternion q = shortestArcQuat(Vec3(0, 1, 0), normal);
btMatrix3x3 m;
m.setRotation(q);
return btQuaternion(m.getColumn(1), heading * DEGREE_TO_RAD) * q;
} // getArenaStartRotation
//-----------------------------------------------------------------------------
/** Loads the drive graph, i.e. the definition of all quads, and the way
* they are connected to each other.
*/
void Track::loadDriveGraph(unsigned int mode_id, const bool reverse)
{
new DriveGraph(m_root+m_all_modes[mode_id].m_quad_name,
m_root+m_all_modes[mode_id].m_graph_name, reverse);
// setGraph is done in DriveGraph constructor
assert(DriveGraph::get());
DriveGraph::get()->setupPaths();
#ifdef DEBUG
for(unsigned int i=0; i<DriveGraph::get()->getNumNodes(); i++)
{
assert(DriveGraph::get()->getNode(i)->getPredecessor(0)!=-1);
}
#endif
if(DriveGraph::get()->getNumNodes()==0)
{
Log::warn("track", "No graph nodes defined for track '%s'\n",
m_filename.c_str());
if (RaceManager::get()->getNumberOfKarts() > 1)
{
Log::fatal("track", "I can handle the lack of driveline in single"
"kart mode, but not with AIs\n");
}
}
else
{
loadMinimap();
}
} // loadDriveGraph
// -----------------------------------------------------------------------------
void Track::mapPoint2MiniMap(const Vec3 &xyz, Vec3 *draw_at) const
{
if (m_minimap_invert_x_z)
{
Vec3 invert = xyz;
invert.setX(-xyz.x());
invert.setZ(-xyz.z());
Graph::get()->mapPoint2MiniMap(invert, draw_at);
}
else
Graph::get()->mapPoint2MiniMap(xyz, draw_at);
draw_at->setX(draw_at->getX() * m_minimap_x_scale);
draw_at->setY(draw_at->getY() * m_minimap_y_scale);
}
// -----------------------------------------------------------------------------
/** Convert the track tree into its physics equivalents.
* \param main_track_count The number of meshes that are already converted
* when the main track was converted. Only the additional meshes
* added later still need to be converted.
* \param for_height_map Ignore physics only objects which can affect
* height map calculation.
*/
void Track::createPhysicsModel(unsigned int main_track_count,
bool for_height_map)
{
// Remove the temporary track rigid body, and then convert all objects
// (i.e. the track and all additional objects) into a new rigid body
// and convert this again. So this way we have an optimised track
// rigid body which includes all track objects.
// Note that removing the rigid body does not remove the already collected
// triangle information, so there is no need to convert the actual track
// (first element in m_track_mesh) again!
if (m_track_mesh == NULL)
{
Log::error("track",
"m_track_mesh == NULL, cannot createPhysicsModel\n");
return;
}
// Now convert all objects that are only used for the physics
// (like invisible walls).
if (!for_height_map)
{
for (unsigned int i = 0; i<m_static_physics_only_nodes.size(); i++)
{
main_loop->renderGUI(5550, i, m_static_physics_only_nodes.size());
convertTrackToBullet(m_static_physics_only_nodes[i]);
if (UserConfigParams::m_physics_debug &&
m_static_physics_only_nodes[i]->getType() == scene::ESNT_MESH)
{
const video::SColor color(255, 255, 105, 180);
scene::IMesh *mesh = ((scene::IMeshSceneNode*)m_static_physics_only_nodes[i])->getMesh();
scene::IMeshBuffer *mb = mesh->getMeshBuffer(0);
mb->getMaterial().BackfaceCulling = false;
video::S3DVertex * const verts = (video::S3DVertex *) mb->getVertices();
const u32 max = mb->getVertexCount();
for (i = 0; i < max; i++)
{
verts[i].Color = color;
}
}
else
irr_driver->removeNode(m_static_physics_only_nodes[i]);
}
main_loop->renderGUI(5560);
if (!UserConfigParams::m_physics_debug)
m_static_physics_only_nodes.clear();
for (unsigned int i = 0; i<m_object_physics_only_nodes.size(); i++)
{
main_loop->renderGUI(5565, i, m_static_physics_only_nodes.size());
convertTrackToBullet(m_object_physics_only_nodes[i]);
m_object_physics_only_nodes[i]->setVisible(false);
m_object_physics_only_nodes[i]->grab();
irr_driver->removeNode(m_object_physics_only_nodes[i]);
}
}
m_track_mesh->removeAll();
if (m_gfx_effect_mesh)
m_gfx_effect_mesh->removeAll();
for(unsigned int i=main_track_count; i<m_all_nodes.size(); i++)
{
main_loop->renderGUI(5570, i, m_all_nodes.size());
convertTrackToBullet(m_all_nodes[i]);
uploadNodeVertexBuffer(m_all_nodes[i]);
}
main_loop->renderGUI(5580);
if (for_height_map)
m_track_mesh->createCollisionShape();
else
m_track_mesh->createPhysicalBody(m_friction);
main_loop->renderGUI(5585);
if (m_gfx_effect_mesh)
m_gfx_effect_mesh->createCollisionShape();
main_loop->renderGUI(5590);
} // createPhysicsModel
// -----------------------------------------------------------------------------
/** Convert the graohics track into its physics equivalents.
* \param mesh The mesh to convert.
* \param node The scene node.
*/
void Track::convertTrackToBullet(scene::ISceneNode *node)
{
if (node->getType() == scene::ESNT_TEXT)
return;
if (node->getType() == scene::ESNT_LOD_NODE)
{
node = ((LODNode*)node)->getFirstNode();
if (node == NULL)
{
Log::warn("track",
"This track contains an empty LOD group.");
return;
}
}
node->updateAbsolutePosition();
std::vector<core::matrix4> matrices;
matrices.push_back(node->getAbsoluteTransformation());
scene::IMesh *mesh;
switch(node->getType())
{
case scene::ESNT_MESH :
case scene::ESNT_WATER_SURFACE :
case scene::ESNT_OCTREE :
mesh = ((scene::IMeshSceneNode*)node)->getMesh();
break;
case scene::ESNT_ANIMATED_MESH :
mesh = ((scene::IAnimatedMeshSceneNode*)node)->getMesh();
break;
case scene::ESNT_SKY_BOX :
case scene::ESNT_PARTICLE_SYSTEM :
case scene::ESNT_TEXT:
// These are non-physical
return;
break;
default:
int type_as_int = node->getType();
char* type = (char*)&type_as_int;
Log::debug("track",
"[convertTrackToBullet] Unknown scene node type : %c%c%c%c.\n",
type[0], type[1], type[2], type[3]);
return;
} // switch node->getType()
//core::matrix4 mat;
//mat.setRotationDegrees(hpr);
//mat.setTranslation(pos);
//core::matrix4 mat_scale;
//// Note that we can't simply call mat.setScale, since this would
//// overwrite the elements on the diagonal, making any rotation incorrect.
//mat_scale.setScale(scale);
//mat *= mat_scale;
for(unsigned int i=0; i<mesh->getMeshBufferCount(); i++)
{
scene::IMeshBuffer *mb = mesh->getMeshBuffer(i);
// FIXME: take translation/rotation into account
if (mb->getVertexType() != video::EVT_STANDARD &&
mb->getVertexType() != video::EVT_2TCOORDS &&
mb->getVertexType() != video::EVT_TANGENTS &&
mb->getVertexType() != video::EVT_SKINNED_MESH)
{
Log::warn("track", "convertTrackToBullet: Ignoring type '%d'!\n",
mb->getVertexType());
continue;
}
u16 *mbIndices = mb->getIndices();
Vec3 vertices[3];
Vec3 normals[3];
#ifndef SERVER_ONLY
SP::SPMeshBuffer* spmb = dynamic_cast<SP::SPMeshBuffer*>(mb);
if (spmb)
{
video::S3DVertexSkinnedMesh* mbVertices = (video::S3DVertexSkinnedMesh*)mb->getVertices();
for (unsigned int matrix_index = 0; matrix_index < matrices.size(); matrix_index++)
{
for (unsigned int j = 0; j < mb->getIndexCount(); j += 3)
{
TriangleMesh* tmesh = m_track_mesh;
Material* material = spmb->getSTKMaterial(j);
if (material->isSurface())
{
tmesh = m_gfx_effect_mesh;
}
else if (material->isIgnore())
{
continue;
}
for (unsigned int k = 0; k < 3; k++)
{
int indx = mbIndices[j + k];
core::vector3df v = mbVertices[indx].m_position;
matrices[matrix_index].transformVect(v);
vertices[k] = v;
normals[k] = MiniGLM::decompressVector3(mbVertices[indx].m_normal);
} // for k
if (tmesh)
{
tmesh->addTriangle(vertices[0], vertices[1],
vertices[2], normals[0],
normals[1], normals[2],
material);
}
} // for j
} // for matrix_index
}
else
#endif
{
const video::SMaterial& irrMaterial = mb->getMaterial();
std::string t1_full_path, t2_full_path;
video::ITexture* t1 = irrMaterial.getTexture(0);
if (t1)
{
t1_full_path = t1->getName().getPtr();
t1_full_path = file_manager->getFileSystem()->getAbsolutePath(
t1_full_path.c_str()).c_str();
}
video::ITexture* t2 = irrMaterial.getTexture(1);
if (t2)
{
t2_full_path = t2->getName().getPtr();
t2_full_path = file_manager->getFileSystem()->getAbsolutePath(
t2_full_path.c_str()).c_str();
}
const Material* material = material_manager->getMaterialSPM(
t1_full_path, t2_full_path);
TriangleMesh *tmesh = m_track_mesh;
// Special gfx meshes will not be stored as a normal physics body,
// but converted to a collision body only, so that ray tests
// against them can be done.
if (material->isSurface())
tmesh = m_gfx_effect_mesh;
// A material which is a surface must be converted,
// even if it's marked as ignore. So only ignore
// non-surface materials.
else if(material->isIgnore())
continue;
if (mb->getVertexType() == video::EVT_STANDARD)
{
irr::video::S3DVertex* mbVertices=(video::S3DVertex*)mb->getVertices();
for (unsigned int matrix_index = 0; matrix_index < matrices.size(); matrix_index++)
{
for (unsigned int j = 0; j < mb->getIndexCount(); j += 3)
{
for (unsigned int k = 0; k < 3; k++)
{
int indx = mbIndices[j + k];
core::vector3df v = mbVertices[indx].Pos;
matrices[matrix_index].transformVect(v);
vertices[k] = v;
normals[k] = mbVertices[indx].Normal;
} // for k
if (tmesh)
{
tmesh->addTriangle(vertices[0], vertices[1],
vertices[2], normals[0],
normals[1], normals[2],
material);
}
} // for j
} // for matrix_index
}
else if (mb->getVertexType() == video::EVT_2TCOORDS)
{
irr::video::S3DVertex2TCoords* mbVertices = (video::S3DVertex2TCoords*)mb->getVertices();
for (unsigned int matrix_index = 0; matrix_index < matrices.size(); matrix_index++)
{
for (unsigned int j = 0; j < mb->getIndexCount(); j += 3)
{
for (unsigned int k = 0; k < 3; k++)
{
int indx = mbIndices[j + k];
core::vector3df v = mbVertices[indx].Pos;
matrices[matrix_index].transformVect(v);
vertices[k] = v;
normals[k] = mbVertices[indx].Normal;
} // for k
if (tmesh)
{
tmesh->addTriangle(vertices[0], vertices[1],
vertices[2], normals[0],
normals[1], normals[2],
material);
}
} // for j
} // for matrix_index
}
else if (mb->getVertexType() == video::EVT_TANGENTS)
{
irr::video::S3DVertexTangents* mbVertices = (video::S3DVertexTangents*)mb->getVertices();
for (unsigned int matrix_index = 0; matrix_index < matrices.size(); matrix_index++)
{
for (unsigned int j = 0; j < mb->getIndexCount(); j += 3)
{
for (unsigned int k = 0; k < 3; k++)
{
int indx = mbIndices[j + k];
core::vector3df v = mbVertices[indx].Pos;
matrices[matrix_index].transformVect(v);
vertices[k] = v;
normals[k] = mbVertices[indx].Normal;
} // for k
if (tmesh)
{
tmesh->addTriangle(vertices[0], vertices[1],
vertices[2], normals[0],
normals[1], normals[2],
material);
}
} // for j
} // for matrix_index
}
else if (mb->getVertexType() == video::EVT_SKINNED_MESH)
{
video::S3DVertexSkinnedMesh* mbVertices = (video::S3DVertexSkinnedMesh*)mb->getVertices();
for (unsigned int matrix_index = 0; matrix_index < matrices.size(); matrix_index++)
{
for (unsigned int j = 0; j < mb->getIndexCount(); j += 3)
{
for (unsigned int k = 0; k < 3; k++)
{
int indx = mbIndices[j + k];
core::vector3df v = mbVertices[indx].m_position;
matrices[matrix_index].transformVect(v);
vertices[k] = v;
normals[k] = MiniGLM::decompressVector3(mbVertices[indx].m_normal);
} // for k
if (tmesh)
{
tmesh->addTriangle(vertices[0], vertices[1],
vertices[2], normals[0],
normals[1], normals[2],
material);
}
} // for j
} // for matrix_index
}
}
} // for i<getMeshBufferCount
} // convertTrackToBullet
// ----------------------------------------------------------------------------
void Track::loadMinimap()
{
#ifndef SERVER_ONLY
if (GUIEngine::isNoGraphics())
return;
//Create the minimap resizing it as necessary.
core::dimension2du mini_map_size = World::getWorld()->getRaceGUI()->getMiniMapSize();
//Use twice the size of the rendered minimap to reduce significantly aliasing
m_render_target = Graph::get()->makeMiniMap(mini_map_size * 2,
"minimap::" + m_ident, video::SColor(127, 255, 255, 255),
m_minimap_invert_x_z);
updateMiniMapScale();
#endif
} // loadMinimap
// ----------------------------------------------------------------------------
void Track::updateMiniMapScale()
{
if (!m_render_target)
return;
core::dimension2du mini_map_size = World::getWorld()->getRaceGUI()->getMiniMapSize();
// Happens in race result gui
if (mini_map_size.Width == 0 || mini_map_size.Height == 0)
return;
core::dimension2du mini_map_texture_size = m_render_target->getTextureSize();
if(mini_map_texture_size.Width)
m_minimap_x_scale = float(mini_map_size.Width) / float(mini_map_texture_size.Width);
else
m_minimap_x_scale = 0;
if(mini_map_texture_size.Height)
m_minimap_y_scale = float(mini_map_size.Height) / float(mini_map_texture_size.Height);
else
m_minimap_y_scale = 0;
}
// ----------------------------------------------------------------------------
/** Loads the main track model (i.e. all other objects contained in the
* scene might use raycast on this track model to determine the actual
* height of the terrain.
*/
bool Track::loadMainTrack(const XMLNode &root)
{
assert(m_track_mesh==NULL);
assert(m_height_map_mesh==NULL);
assert(m_gfx_effect_mesh==NULL);
m_challenges.clear();
m_track_mesh = new TriangleMesh(/*can_be_transformed*/false);
m_gfx_effect_mesh = new TriangleMesh(/*can_be_transformed*/false);
const XMLNode *track_node = root.getNode("track");
std::string model_name;
track_node->get("model", &model_name);
std::string full_path = m_root+model_name;
scene::IMesh *mesh = irr_driver->getMesh(full_path);
if(!mesh)
{
Log::fatal("track",
"Main track model '%s' in '%s' not found, aborting.\n",
track_node->getName().c_str(), model_name.c_str());
}
scene::IAnimatedMesh* an_mesh = dynamic_cast<scene::IAnimatedMesh*>(mesh);
bool ge_spm = false;
if (an_mesh && an_mesh->getMeshType() == scene::EAMT_SPM)
ge_spm = true;
scene::ISceneNode* scene_node = NULL;
scene::IMesh* tangent_mesh = NULL;
#ifdef SERVER_ONLY
if (false)
#else
if (m_version < 7 && !CVS->isGLSL() && !GUIEngine::isNoGraphics() &&
!ge_spm)
#endif
{
// The mesh as returned does not have all mesh buffers with the same
// texture combined. This can result in a _HUGE_ overhead. E.g. instead
// of 46 different mesh buffers over 500 (for some tracks even >1000)
// were created. This means less effect from hardware support, less
// vertices per opengl operation, more overhead on CPU, ...
// So till we have a better b3d exporter which can combine the different
// meshes which use the same texture when exporting, the meshes are
// combined using CBatchingMesh.
scene::CBatchingMesh *merged_mesh = new scene::CBatchingMesh();
merged_mesh->addMesh(mesh);
merged_mesh->finalize();
tangent_mesh = merged_mesh;
// The reference count of the mesh is 1, since it is in irrlicht's
// cache. So we only have to remove it from the cache.
irr_driver->removeMeshFromCache(mesh);
}
else
{
// SPM does the combine for you
tangent_mesh = mesh;
tangent_mesh->grab();
}
// The merged mesh is grabbed by the octtree, so we don't need
// to keep a reference to it.
scene_node = irr_driver->addMesh(tangent_mesh, "track_main");
// We should drop the merged mesh (since it's now referred to in the
// scene node), but then we need to grab it since it's in the
// m_all_cached_meshes.
m_all_cached_meshes.push_back(tangent_mesh);
irr_driver->grabAllTextures(tangent_mesh);
main_loop->renderGUI(4000);
#ifdef DEBUG
std::string debug_name=model_name+" (main track, octtree)";
scene_node->setName(debug_name.c_str());
#endif
//merged_mesh->setHardwareMappingHint(scene::EHM_STATIC);
core::vector3df xyz(0,0,0);
track_node->getXYZ(&xyz);
core::vector3df hpr(0,0,0);
track_node->getHPR(&hpr);
scene_node->setPosition(xyz);
scene_node->setRotation(hpr);
handleAnimatedTextures(scene_node, *track_node);
m_all_nodes.push_back(scene_node);
MeshTools::minMax3D(tangent_mesh, &m_aabb_min, &m_aabb_max);
// Increase the maximum height of the track: since items that fly
// too high explode, e.g. cakes can not be show when being at the
// top of the track (since they will explode when leaving the AABB
// of the track). While the test for this in Flyable::updateAndDelete
// could be relaxed to fix this, it is not certain how the physics
// will handle items that are out of the AABB
m_aabb_max.setY(m_aabb_max.getY()+30.0f);
Physics::get()->init(m_aabb_min, m_aabb_max);
ModelDefinitionLoader lodLoader(this);
// Load LOD groups
const XMLNode *lod_xml_node = root.getNode("lod");
if (lod_xml_node != NULL)
{
for (unsigned int i = 0; i < lod_xml_node->getNumNodes(); i++)
{
main_loop->renderGUI(4100, i, lod_xml_node->getNumNodes());
const XMLNode* lod_group_xml = lod_xml_node->getNode(i);
for (unsigned int j = 0; j < lod_group_xml->getNumNodes(); j++)
{
lodLoader.addModelDefinition(lod_group_xml->getNode(j));
}
}
}
main_loop->renderGUI(4200);
for (unsigned int i=0; i<track_node->getNumNodes(); i++)
{
main_loop->renderGUI(4300, i, track_node->getNumNodes());
const XMLNode *n=track_node->getNode(i);
// Animated textures have already been handled
if(n->getName()=="animated-texture") continue;
// Only "object" entries are allowed now inside of the model tag
if(n->getName()!="static-object")
{
Log::error("track",
"Incorrect tag '%s' inside <model> of scene file - ignored\n",
n->getName().c_str());
continue;
}
core::vector3df xyz(0,0,0);
n->get("xyz", &xyz);
core::vector3df hpr(0,0,0);
n->get("hpr", &hpr);
core::vector3df scale(1.0f, 1.0f, 1.0f);
n->get("scale", &scale);
bool tangent = false;
n->get("tangents", &tangent);
scene::ISceneNode* scene_node;
model_name="";
n->get("model", &model_name);
full_path = m_root+model_name;
std::string interaction;
n->get("interaction", &interaction);
// a special challenge orb object for overworld
std::string challenge;
n->get("challenge", &challenge);
bool lod_instance = false;
n->get("lod_instance", &lod_instance);
if (lod_instance)
{
LODNode* node = lodLoader.instanciateAsLOD(n, NULL, NULL);
if (node != NULL)
{
node->setPosition(xyz);
node->setRotation(hpr);
node->setScale(scale);
node->updateAbsolutePosition();
m_all_nodes.push_back( node );
}
}
else
{
// TODO: check if mesh is animated or not
scene::IMesh *a_mesh = irr_driver->getMesh(full_path);
if(!a_mesh)
{
Log::error("track", "Object model '%s' not found, ignored.\n",
full_path.c_str());
continue;
}
// The meshes loaded here are in irrlicht's mesh cache. So we
// have to keep track of them in order to properly remove them
// from memory. We could add each track only once in a list, but
// it's actually faster to add meshes multipl times (if they are
// used more than once), and increase the ref count each time.
// When removing the meshes, we drop them till the ref count is
// 1 - which means that the only reference is now in the cache,
// and can therefore be removed.
m_all_cached_meshes.push_back(a_mesh);
irr_driver->grabAllTextures(a_mesh);
a_mesh->grab();
scene_node = irr_driver->addMesh(a_mesh, model_name);
scene_node->setPosition(xyz);
scene_node->setRotation(hpr);
scene_node->setScale(scale);
#ifdef DEBUG
std::string debug_name = model_name+" (static track-object)";
scene_node->setName(debug_name.c_str());
#endif
handleAnimatedTextures(scene_node, *n);
// for challenge orbs, a bit more work to do
// TODO: this is hardcoded for the overworld, convert to scripting
if (challenge.size() > 0)
{
const ChallengeData* c = NULL;
if (challenge != "tutorial")
{
c = unlock_manager->getChallengeData(challenge);
if (c == NULL)
{
Log::error("track", "Cannot find challenge named <%s>\n",
challenge.c_str());
scene_node->remove();
continue;
}
}
m_challenges.push_back( OverworldChallenge(xyz, challenge) );
if (c && c->isGrandPrix())
{
}
else
{
if (challenge != "tutorial")
{
Track* t = track_manager->getTrack(c->getTrackId());
if (t == NULL)
{
Log::error("track", "Cannot find track named <%s>\n",
c->getTrackId().c_str());
continue;
}
std::string sshot = t->getScreenshotFile();
video::ITexture* screenshot = irr_driver->getTexture(sshot);
if (screenshot == NULL)
{
Log::error("track",
"Cannot find track screenshot <%s>",
sshot.c_str());
continue;
}
scene_node->getMaterial(0).setTexture(0, screenshot);
}
}
// make transparent
for (unsigned int m=0; m<a_mesh->getMeshBufferCount(); m++)
{
scene::IMeshBuffer* mb = a_mesh->getMeshBuffer(m);
if (mb->getVertexType() == video::EVT_STANDARD)
{
video::S3DVertex* v = (video::S3DVertex*)mb->getVertices();
for (unsigned int n=0; n<mb->getVertexCount(); n++)
{
v[n].Color.setAlpha(125);
}
}
}
LODNode* lod_node = new LODNode("challenge_orb",
irr_driver->getSceneManager()->getRootSceneNode(),
irr_driver->getSceneManager());
lod_node->setPosition(xyz);
lod_node->add(50, scene_node, true /* reparent */);
m_all_nodes.push_back( lod_node );
}
else
{
if(interaction=="physics-only")
m_static_physics_only_nodes.push_back(scene_node);
else
m_all_nodes.push_back( scene_node );
}
}
} // for i
// This will (at this stage) only convert the main track model.
for(unsigned int i=0; i<m_all_nodes.size(); i++)
{
main_loop->renderGUI(4350, i, m_all_nodes.size());
convertTrackToBullet(m_all_nodes[i]);
main_loop->renderGUI(4360, i, m_all_nodes.size());
uploadNodeVertexBuffer(m_all_nodes[i]);
main_loop->renderGUI(4400, i, m_all_nodes.size());
}
// Free the tangent (track mesh) after converting to physics
if (GUIEngine::isNoGraphics())
tangent_mesh->freeMeshVertexBuffer();
if (m_track_mesh == NULL)
{
Log::fatal("track", "m_track_mesh == NULL, cannot loadMainTrack\n");
}
m_gfx_effect_mesh->createCollisionShape();
scene_node->setMaterialFlag(video::EMF_LIGHTING, true);
scene_node->setMaterialFlag(video::EMF_GOURAUD_SHADING, true);
main_loop->renderGUI(4500);
return true;
} // loadMainTrack
// ----------------------------------------------------------------------------
void Track::freeCachedMeshVertexBuffer()
{
if (GUIEngine::isNoGraphics())
{
for (unsigned i = 0; i < m_all_cached_meshes.size(); i++)
m_all_cached_meshes[i]->freeMeshVertexBuffer();
}
} // freeCachedMeshVertexBuffer
// ----------------------------------------------------------------------------
/** Handles animated textures.
* \param node The scene node for which animated textures are handled.
* \param xml The node containing the data for the animated notion.
*/
void Track::handleAnimatedTextures(scene::ISceneNode *node, const XMLNode &xml)
{
for(unsigned int node_number = 0; node_number<xml.getNumNodes();
node_number++)
{
const XMLNode *texture_node = xml.getNode(node_number);
if(texture_node->getName()!="animated-texture") continue;
std::string name;
texture_node->get("name", &name);
if(name=="")
{
Log::error("track",
"Animated texture: no texture name specified for track '%s'\n",
m_ident.c_str());
continue;
}
// to lower case, for case-insensitive comparison
name = StringUtils::toLowerCase(name);
int moving_textures_found = 0;
SP::SPMeshNode* spmn = dynamic_cast<SP::SPMeshNode*>(node);
if (spmn)
{
for (unsigned i = 0; i < spmn->getSPM()->getMeshBufferCount(); i++)
{
SP::SPMeshBuffer* spmb = spmn->getSPM()->getSPMeshBuffer(i);
const std::vector<Material*>& m = spmb->getAllSTKMaterials();
bool found = false;
for (unsigned j = 0; j < m.size(); j++)
{
Material* mat = m[j];
std::string mat_name =
StringUtils::getBasename(mat->getSamplerPath(0));
mat_name = StringUtils::toLowerCase(mat_name);
if (mat_name == name)
{
found = true;
moving_textures_found++;
spmb->enableTextureMatrix(j);
MovingTexture* mt =
new MovingTexture(NULL, *texture_node);
mt->setSPTM(spmn->getTextureMatrix(i).data());
m_animated_textures.push_back(mt);
// For spm only 1 texture matrix per mesh buffer is
// possible
break;
}
}
if (found)
{
break;
}
}
}
else
{
for(unsigned int i=0; i<node->getMaterialCount(); i++)
{
video::SMaterial &irrMaterial=node->getMaterial(i);
for(unsigned int j=0; j<video::MATERIAL_MAX_TEXTURES; j++)
{
video::ITexture* t=irrMaterial.getTexture(j);
if(!t) continue;
std::string texture_name =
StringUtils::getBasename(t->getName().getPtr());
// to lower case, for case-insensitive comparison
texture_name = StringUtils::toLowerCase(texture_name);
if (texture_name != name) continue;
core::matrix4 *m = &irrMaterial.getTextureMatrix(j);
m_animated_textures.push_back(new MovingTexture(m, *texture_node));
moving_textures_found++;
} // for j<MATERIAL_MAX_TEXTURES
} // for i<getMaterialCount
}
if (moving_textures_found == 0)
Log::warn("AnimTexture", "Did not find animate texture '%s'", name.c_str());
} // for node_number < xml->getNumNodes
} // handleAnimatedTextures
// ----------------------------------------------------------------------------
/** This updates all only graphical elements. It is only called once per
* rendered frame, not once per time step.
* float dt Time since last rame.
*/
void Track::updateGraphics(float dt)
{
m_track_object_manager->updateGraphics(dt);
for (unsigned int i = 0; i<m_animated_textures.size(); i++)
{
m_animated_textures[i]->update(dt);
}
m_item_manager->updateGraphics(dt);
} // updateGraphics
// ----------------------------------------------------------------------------
/** Update, called once per physics time step.
* \param dt Timestep.
*/
void Track::update(int ticks)
{
ProcessType type = STKProcess::getType();
if (type == PT_MAIN && !m_startup_run) // first time running update = good point to run startup script
{
Scripting::ScriptEngine::getInstance()->runFunction(false, "void onStart()");
m_startup_run = true;
// After onStart all track objects will be hidden as needed
// we only copy track objects with physical body which affects network
if (LobbyProtocol::getByType<LobbyProtocol>(PT_CHILD))
{
Track* child_track = clone();
m_current_track[PT_CHILD] = child_track;
}
}
float dt = stk_config->ticks2Time(ticks);
m_check_manager->update(dt);
m_item_manager->update(ticks);
// TODO: enable onUpdate scripts if we ever find a compelling use for them
//Scripting::ScriptEngine* script_engine = World::getWorld()->getScriptEngine();
//script_engine->runScript("void onUpdate()");
} // update
// ----------------------------------------------------------------------------
/** Handles an explosion, i.e. it makes sure that all physical objects are
* affected accordingly.
* \param pos Position of the explosion.
* \param obj If the hit was a physical object, this object will be affected
* more. Otherwise this is NULL.
* \param secondary_hits True if items that are not directly hit should
* also be affected. */
void Track::handleExplosion(const Vec3 &pos, const PhysicalObject *obj,
bool secondary_hits) const
{
m_track_object_manager->handleExplosion(pos, obj, secondary_hits);
} // handleExplosion
// ----------------------------------------------------------------------------
/** Creates a water node. OBSOLETE, kept for backwards compat only
* \param node The XML node containing the specifications for the water node.
*/
void Track::createWater(const XMLNode &node)
{
std::string model_name;
node.get("model", &model_name);
std::string full_path = m_root+model_name;
scene::IMesh *mesh = irr_driver->getMesh(full_path);
if (mesh == NULL)
{
Log::warn("Track", "Water not found : '%s'", full_path.c_str());
return;
}
/*
float wave_height = 2.0f;
float wave_speed = 300.0f;
float wave_length = 10.0f;
node.get("height", &wave_height);
float time;
if(node.get("time", &time))
{
wave_speed = time * 1000.0f/(2.0f*M_PI);
}
else
node.get("speed", &wave_speed);
if(wave_speed==0)
{
// A speed of 0 results in a division by zero, so avoid this.
// The actual time for a wave from one maximum to the next is
// given by 2*M_PI*speed/1000.
Log::warn("Track",
"Wave-speed or time is 0, resetting it to the default.");
wave_speed =300.0f;
}
node.get("length", &wave_length);
*/
scene::ISceneNode* scene_node = NULL;
/*
if (UserConfigParams::m_particles_effects > 1)
{
scene::IMesh *welded;
scene_node = irr_driver->addWaterNode(mesh, &welded,
wave_height,
wave_speed,
wave_length);
mesh->grab();
irr_driver->grabAllTextures(mesh);
m_all_cached_meshes.push_back(mesh);
mesh = welded;
}
else
{*/
scene_node = irr_driver->addMesh(mesh, "water");
//}
if(!mesh || !scene_node)
{
Log::error("track", "Water model '%s' in '%s' not found, ignored.\n",
node.getName().c_str(), model_name.c_str());
return;
}
#ifdef DEBUG
std::string debug_name = model_name+"(water node)";
scene_node->setName(debug_name.c_str());
#endif
mesh->grab();
m_all_cached_meshes.push_back(mesh);
irr_driver->grabAllTextures(mesh);
core::vector3df xyz(0,0,0);
node.get("xyz", &xyz);
core::vector3df hpr(0,0,0);
node.get("hpr", &hpr);
scene_node->setPosition(xyz);
scene_node->setRotation(hpr);
m_all_nodes.push_back(scene_node);
handleAnimatedTextures(scene_node, node);
scene_node->getMaterial(0).setFlag(video::EMF_GOURAUD_SHADING, true);
} // createWater
// ----------------------------------------------------------------------------
static void recursiveUpdatePosition(scene::ISceneNode *node)
{
node->updateAbsolutePosition();
for (unsigned i = 0; i < node->getChildren().size(); i++)
recursiveUpdatePosition(node->getChildren()[i]);
} // recursiveUpdatePosition
// ----------------------------------------------------------------------------
static void recursiveUpdatePhysics(std::vector<TrackObject*>& tos)
{
for (TrackObject* to : tos)
{
if (to->getPhysicalObject())
{
TrackObjectPresentationSceneNode* sn = to
->getPresentation<TrackObjectPresentationSceneNode>();
if (sn)
{
to->getPhysicalObject()->move(
sn->getNode()->getAbsoluteTransformation().getTranslation(),
sn->getNode()->getAbsoluteTransformation()
.getRotationDegrees());
}
}
recursiveUpdatePhysics(to->getChildren());
}
} // recursiveUpdatePhysics
// ----------------------------------------------------------------------------
/** This function load the actual scene, i.e. all parts of the track,
* animations, items, ... It is called from world during initialisation.
* Track is the first model to be loaded, so at this stage the root scene node
* is empty.
* \param parent The actual world.
* \param reverse_track True if the track should be run in reverse.
* \param mode_id Which of the modes of a track to use. This determines which
* scene, quad, and graph file to load.
*/
void Track::loadTrackModel(bool reverse_track, unsigned int mode_id)
{
assert(m_current_track[PT_MAIN].load() == NULL);
// Use m_filename to also get the path, not only the identifier
STKTexManager::getInstance()
->setTextureErrorMessage("While loading track '%s'", m_filename);
if(!m_reverse_available)
{
reverse_track = false;
}
main_loop->renderGUI(3000);
m_check_manager = new CheckManager();
assert(m_all_cached_meshes.size()==0);
if(UserConfigParams::logMemory())
{
Log::debug("[memory] Before loading '%s': mesh cache %d "
"texture cache %d\n",
getIdent().c_str(),
irr_driver->getSceneManager()->getMeshCache()->getMeshCount(),
irr_driver->getVideoDriver()->getTextureCount());
#ifdef DEBUG
scene::IMeshCache *cache =
irr_driver->getSceneManager()->getMeshCache();
m_old_mesh_buffers.clear();
for(unsigned int i=0; i<cache->getMeshCount(); i++)
{
const io::SNamedPath &name=cache->getMeshName(i);
m_old_mesh_buffers.push_back(name.getInternalName().c_str());
}
m_old_textures.clear();
video::IVideoDriver *vd = irr_driver->getVideoDriver();
for(unsigned int i=0; i<vd->getTextureCount(); i++)
{
video::ITexture *t=vd->getTextureByIndex(i);
m_old_textures.push_back(t);
}
#endif
}
CameraEnd::clearEndCameras();
m_minimap_invert_x_z = false;
m_sky_type = SKY_NONE;
m_track_object_manager = new TrackObjectManager();
std::string unique_id = StringUtils::insertValues("tracks/%s", m_ident.c_str());
// Add the track directory to the texture search path
file_manager->pushTextureSearchPath(m_root, unique_id);
file_manager->pushModelSearchPath(m_root);
main_loop->renderGUI(3100);
#ifndef SERVER_ONLY
if (CVS->isGLSL())
{
SP::SPShaderManager::get()->loadSPShaders(m_root);
}
#endif
main_loop->renderGUI(3200);
// First read the temporary materials.xml file if it exists
try
{
std::string materials_file = m_root+"materials.xml";
if(m_cache_track)
{
if(!m_materials_loaded)
material_manager->addSharedMaterial(materials_file);
m_materials_loaded = true;
}
else
material_manager->pushTempMaterial(materials_file);
}
catch (std::exception& e)
{
// no temporary materials.xml file, ignore
(void)e;
}
main_loop->renderGUI(3300);
// Start building the scene graph
// Soccer field with navmesh requires it
// for two goal line to be drawn them in minimap
std::string path = m_root + m_all_modes[mode_id].m_scene;
XMLNode *root = file_manager->createXMLTree(path);
// Make sure that we have a track (which is used for raycasts to
// place other objects).
if (!root || root->getName()!="scene")
{
std::ostringstream msg;
msg<< "No track model defined in '"<<path
<<"', aborting.";
throw std::runtime_error(msg.str());
}
m_current_track[PT_MAIN] = this;
m_current_track[PT_CHILD] = NULL;
// Load the graph only now: this function is called from world, after
// the race gui was created. The race gui is needed since it stores
// the information about the size of the texture to render the mini
// map to.
// Load the un-raycasted flag position first (for minimap)
if (m_is_ctf && RaceManager::get()->isCTFMode())
{
for (unsigned int i=0; i<root->getNumNodes(); i++)
{
const XMLNode *node = root->getNode(i);
const std::string &name = node->getName();
if (name == "red-flag")
{
m_red_flag.setOrigin(flagCommand(node));
}
else if (name == "blue-flag")
{
m_blue_flag.setOrigin(flagCommand(node));
}
} // for i<root->getNumNodes()
}
main_loop->renderGUI(3320);
if (!m_is_arena && !m_is_soccer && !m_is_cutscene)
loadDriveGraph(mode_id, reverse_track);
else if ((m_is_arena || m_is_soccer) && !m_is_cutscene && m_has_navmesh)
loadArenaGraph(*root);
main_loop->renderGUI(3340);
if (NetworkConfig::get()->isNetworking())
{
auto nim = std::make_shared<NetworkItemManager>();
nim->rewinderAdd();
m_item_manager = nim;
}
else
{
// Seed random engine locally
uint32_t seed = (uint32_t)StkTime::getTimeSinceEpoch();
ItemManager::updateRandomSeed(seed);
m_item_manager = std::make_shared<ItemManager>();
powerup_manager->setRandomSeed(seed);
}
main_loop->renderGUI(3360);
// Set the default start positions. Node that later the default
// positions can still be overwritten.
float forwards_distance = 1.5f;
float sidewards_distance = 3.0f;
float upwards_distance = 0.1f;
int karts_per_row = 2;
const XMLNode *default_start = root->getNode("default-start");
if (default_start)
{
default_start->get("forwards-distance", &forwards_distance );
default_start->get("sidewards-distance", &sidewards_distance);
default_start->get("upwards-distance", &upwards_distance );
default_start->get("karts-per-row", &karts_per_row );
}
if (!m_is_arena && !m_is_soccer && !m_is_cutscene)
{
if (RaceManager::get()->isFollowMode())
{
// In a FTL race the non-leader karts are placed at the end of the
// field, so we need all start positions.
m_start_transforms.resize(stk_config->m_max_karts);
}
else
m_start_transforms.resize(RaceManager::get()->getNumberOfKarts());
DriveGraph::get()->setDefaultStartPositions(&m_start_transforms,
karts_per_row,
forwards_distance,
sidewards_distance,
upwards_distance);
}
main_loop->renderGUI(3400);
// we need to check for fog before loading the main track model
if (const XMLNode *node = root->getNode("sun"))
{
node->get("xyz", &m_sun_position );
node->get("ambient", &m_default_ambient_color);
node->get("sun-specular", &m_sun_specular_color);
node->get("sun-diffuse", &m_sun_diffuse_color);
node->get("fog", &m_use_fog);
node->get("fog-color", &m_fog_color);
node->get("fog-max", &m_fog_max);
node->get("fog-start", &m_fog_start);
node->get("fog-end", &m_fog_end);
node->get("fog-start-height", &m_fog_height_start);
node->get("fog-end-height", &m_fog_height_end);
}
main_loop->renderGUI(3500);
#ifndef SERVER_ONLY
if (!GUIEngine::isNoGraphics() && CVS->isGLSL() && m_use_fog)
{
glBindBuffer(GL_UNIFORM_BUFFER, SP::sp_fog_ubo);
glBufferSubData(GL_UNIFORM_BUFFER, 0, 4, &m_fog_start);
glBufferSubData(GL_UNIFORM_BUFFER, 4, 4, &m_fog_end);
glBufferSubData(GL_UNIFORM_BUFFER, 8, 4, &m_fog_max);
// Fog density
float val = -(1.0f / (40.0f * (m_fog_start + 0.001f)));
glBufferSubData(GL_UNIFORM_BUFFER, 12, 4, &val);
val = (float)m_fog_color.getRed() / 255.0f;
glBufferSubData(GL_UNIFORM_BUFFER, 16, 4, &val);
val = (float)m_fog_color.getGreen() / 255.0f;
glBufferSubData(GL_UNIFORM_BUFFER, 20, 4, &val);
val = (float)m_fog_color.getBlue() / 255.0f;
glBufferSubData(GL_UNIFORM_BUFFER, 24, 4, &val);
val = 0.0f;
glBufferSubData(GL_UNIFORM_BUFFER, 28, 4, &val);
glBindBuffer(GL_UNIFORM_BUFFER, 0);
}
else if (CVS->isGLSL())
{
SP::resetEmptyFogColor();
}
#endif
main_loop->renderGUI(3600);
if (const XMLNode *node = root->getNode("lightshaft"))
{
m_godrays = true;
node->get("opacity", &m_godrays_opacity);
node->get("color", &m_godrays_color);
node->get("xyz", &m_godrays_position);
}
loadMainTrack(*root);
main_loop->renderGUI(4700);
unsigned int main_track_count = (unsigned int)m_all_nodes.size();
ModelDefinitionLoader model_def_loader(this);
main_loop->renderGUI(4800);
// Load LOD groups
const XMLNode *lod_xml_node = root->getNode("lod");
if (lod_xml_node != NULL)
{
for (unsigned int i = 0; i < lod_xml_node->getNumNodes(); i++)
{
main_loop->renderGUI(4900, i, lod_xml_node->getNumNodes());
const XMLNode* lod_group_xml = lod_xml_node->getNode(i);
for (unsigned int j = 0; j < lod_group_xml->getNumNodes(); j++)
{
model_def_loader.addModelDefinition(lod_group_xml->getNode(j));
}
}
}
loadObjects(root, path, model_def_loader, true, NULL, NULL);
main_loop->renderGUI(5000);
// If the track is low complexity, increase distances for LoD nodes
// Scene complexity is not computed before LoD nodes are loaded, so
// instead we set a variable that will be used to update the distances
// later on.
float squared_multiplier = 1.0f;
if (irr_driver->getSceneComplexity() < 1500)
{
float ratio = 1.0f;
// Cap the potential effect
if (irr_driver->getSceneComplexity() < 100)
ratio = 15.0f;
else
ratio = 1500.0f / (float)(irr_driver->getSceneComplexity());
squared_multiplier = 0.3f + 0.7f * ratio;
}
irr_driver->setLODMultiplier(squared_multiplier);
// The LoD distances are stored squared in the node, therefore the real multiplier
// is the square root of the one that gets applied
Log::info("Track", "Overall scene complexity estimated at %d, Auto-LoD multiplier is %f",
irr_driver->getSceneComplexity(), sqrtf(squared_multiplier));
// Correct the parenting of meta library
for (auto& p : m_meta_library)
{
auto* ln = p.first->getPresentation<TrackObjectPresentationLibraryNode>();
assert(ln);
TrackObjectPresentationLibraryNode* meta_ln = p.second
->getPresentation<TrackObjectPresentationLibraryNode>();
assert(meta_ln);
meta_ln->getNode()->setParent(ln->getNode());
recursiveUpdatePosition(meta_ln->getNode());
recursiveUpdatePhysics(p.second->getChildren());
main_loop->renderGUI(5050);
}
model_def_loader.cleanLibraryNodesAfterLoad();
main_loop->renderGUI(5100);
Scripting::ScriptEngine::getInstance()->compileLoadedScripts();
main_loop->renderGUI(5200);
// Init all track objects
m_track_object_manager->init();
main_loop->renderGUI(5300);
// ---- Fog
// It's important to execute this BEFORE the code that creates the skycube,
// otherwise the skycube node could be modified to have fog enabled, which
// we don't want
#ifndef SERVER_ONLY
if (m_use_fog && Camera::getDefaultCameraType()!=Camera::CM_TYPE_DEBUG &&
!CVS->isGLSL())
{
/* NOTE: if LINEAR type, density does not matter, if EXP or EXP2, start
and end do not matter */
irr_driver->getVideoDriver()->setFog(m_fog_color,
video::EFT_FOG_LINEAR,
m_fog_start, m_fog_end,
1.0f);
}
#endif
// Sky dome and boxes support
// --------------------------
irr_driver->suppressSkyBox();
#ifndef SERVER_ONLY
if(m_sky_type==SKY_BOX && m_sky_textures.size() == 6)
{
if (CVS->isGLSL())
{
std::vector<video::IImage*> sky;
for (void* t : m_sky_textures)
sky.push_back((video::IImage*)t);
std::vector<video::IImage*> sh;
for (void* t : m_spherical_harmonics_textures)
sh.push_back((video::IImage*)t);
SP::getRenderer()->addSkyBox(sky, sh);
}
else
{
std::vector<video::ITexture*> textures;
for (void* t : m_sky_textures)
textures.push_back((video::ITexture*)t);
m_all_nodes.push_back(irr_driver->addSkyBox(textures, {}));
}
}
else if(m_sky_type==SKY_COLOR)
{
irr_driver->setClearbackBufferColor(m_sky_color);
}
#endif
main_loop->renderGUI(5400);
// ---- Set ambient color
m_ambient_color = m_default_ambient_color;
irr_driver->setAmbientLight(m_ambient_color,
m_spherical_harmonics_textures.size() != 6/*force_SH_computation*/);
// ---- Create sun (non-ambient directional light)
if (m_sun_position.getLengthSQ() < 0.03f)
{
m_sun_position = core::vector3df(500, 250, 250);
}
const video::SColorf tmpf(m_sun_diffuse_color);
m_sun = irr_driver->addLight(m_sun_position, 0., 0., tmpf.r, tmpf.g, tmpf.b, true);
#ifndef SERVER_ONLY
if (!CVS->isGLSL())
{
scene::ILightSceneNode *sun_ = (scene::ILightSceneNode *) m_sun;
sun_->setLightType(video::ELT_DIRECTIONAL);
// The angle of the light is rather important - let the sun
// point towards (0,0,0).
if (m_sun_position.getLengthSQ() < 0.03f)
// Backward compatibility: if no sun is specified, use the
// old hardcoded default angle
m_sun->setRotation(core::vector3df(180, 45, 45));
else
m_sun->setRotation((-m_sun_position).getHorizontalAngle());
sun_->getLightData().SpecularColor = m_sun_specular_color;
}
else
{
irr_driver->createSunInterposer();
m_sun->grab();
}
#endif
main_loop->renderGUI(5500);
// Join all static physics only object to main track if possible
// Take the visibility condition by scripting into account
std::vector<TrackObject*> objs_removing;
for (auto* to : m_track_object_manager->getObjects().m_contents_vector)
{
if (to->joinToMainTrack())
{
m_track_object_manager->removeDriveableObject(to);
TrackObjectPresentationSceneNode* ts =
to->getPresentation<TrackObjectPresentationSceneNode>();
// physicial only node is always hidden, remove it from stk after
// joining to track mesh
if (ts && ts->isAlwaysHidden())
objs_removing.push_back(to);
}
}
for (auto* obj : objs_removing)
m_track_object_manager->removeObject(obj);
if (!GUIEngine::isNoGraphics())
{
m_height_map_mesh = new TriangleMesh(/*can_be_transformed*/false);
m_height_map_mesh->copyFrom(*m_track_mesh);
TriangleMesh* gfx_effect_mesh = m_gfx_effect_mesh;
std::swap(m_track_mesh, m_height_map_mesh);
m_gfx_effect_mesh = NULL;
createPhysicsModel(main_track_count, true/*for_height_map*/);
std::swap(m_track_mesh, m_height_map_mesh);
std::swap(m_gfx_effect_mesh, gfx_effect_mesh);
}
createPhysicsModel(main_track_count, false/*for_height_map*/);
main_loop->renderGUI(5600);
freeCachedMeshVertexBuffer();
const bool arena_random_item_created =
m_item_manager->randomItemsForArena(m_start_transforms);
if (!arena_random_item_created)
{
for (unsigned int i=0; i<root->getNumNodes(); i++)
{
const XMLNode *node = root->getNode(i);
const std::string &name = node->getName();
if (name=="banana" || name=="item" ||
name=="small-nitro" || name=="big-nitro" ||
name=="easter-egg" )
{
itemCommand(node);
}
} // for i<root->getNumNodes()
}
main_loop->renderGUI(5700);
if (m_is_ctf && RaceManager::get()->isCTFMode())
{
for (unsigned int i=0; i<root->getNumNodes(); i++)
{
const XMLNode *node = root->getNode(i);
const std::string &name = node->getName();
if (name == "red-flag" || name == "blue-flag")
{
flagCommand(node);
}
} // for i<root->getNumNodes()
}
delete root;
main_loop->renderGUI(5800);
if (auto sl = LobbyProtocol::get<ServerLobby>())
{
sl->saveInitialItems(
std::dynamic_pointer_cast<NetworkItemManager>(m_item_manager));
}
main_loop->renderGUI(5900);
if (UserConfigParams::m_track_debug && Graph::get() && !m_is_cutscene)
Graph::get()->createDebugMesh();
// Only print warning if not in battle mode, since battle tracks don't have
// any quads or check lines.
if (m_check_manager->getCheckStructureCount()==0 &&
!RaceManager::get()->isBattleMode() && !m_is_cutscene)
{
Log::warn("track", "No check lines found in track '%s'.",
m_ident.c_str());
Log::warn("track", "Lap counting will not work, and start "
"positions might be incorrect.");
}
if (UserConfigParams::logMemory())
{
Log::debug("track", "[memory] After loading '%s': mesh cache %d "
"texture cache %d\n", getIdent().c_str(),
irr_driver->getSceneManager()->getMeshCache()->getMeshCount(),
irr_driver->getVideoDriver()->getTextureCount());
}
World *world = World::getWorld();
if (world->useChecklineRequirements())
{
DriveGraph::get()->computeChecklineRequirements();
}
main_loop->renderGUI(6000);
EasterEggHunt *easter_world = dynamic_cast<EasterEggHunt*>(world);
if(easter_world)
{
std::string dir = StringUtils::getPath(m_filename);
easter_world->readData(dir+"/easter_eggs.xml");
}
main_loop->renderGUI(6100);
STKTexManager::getInstance()->unsetTextureErrorMessage();
#ifndef SERVER_ONLY
if (CVS->isGLSL())
{
m_sky_textures.clear();
m_spherical_harmonics_textures.clear();
}
#endif // !SERVER_ONLY
} // loadTrackModel
//-----------------------------------------------------------------------------
void Track::loadObjects(const XMLNode* root, const std::string& path,
ModelDefinitionLoader& model_def_loader,
bool create_lod_definitions, scene::ISceneNode* parent,
TrackObject* parent_library)
{
unsigned int start_position_counter = 0;
unsigned int node_count = root->getNumNodes();
const bool is_mode_ctf = m_is_ctf && RaceManager::get()->isCTFMode();
// We keep track of the complexity of the scene (amount of objects loaded, etc)
irr_driver->addSceneComplexity(node_count);
for (unsigned int i = 0; i < node_count; i++)
{
main_loop->renderGUI(4950, i, node_count);
const XMLNode *node = root->getNode(i);
const std::string name = node->getName();
// The track object was already converted before the loop, and the
// default start was already used, too - so ignore those.
if (name == "track" || name == "default-start") continue;
if (name == "object" || name == "library")
{
m_track_object_manager->add(*node, parent, model_def_loader, parent_library);
}
else if (name == "water")
{
createWater(*node);
}
else if (name == "banana" || name == "item" ||
name == "small-nitro" || name == "big-nitro" ||
name == "easter-egg" || name == "red-flag" ||
name == "blue-flag")
{
// will be handled later
}
else if (name == "start" || name == "ctf-start")
{
if ((name == "start" && is_mode_ctf) ||
(name == "ctf-start" && !is_mode_ctf))
continue;
unsigned int position = start_position_counter;
start_position_counter++;
node->get("position", &position);
Vec3 xyz(0,0,0);
node->getXYZ(&xyz);
float h=0;
node->get("h", &h);
if (position >= m_start_transforms.size())
{
m_start_transforms.resize(position + 1);
}
m_start_transforms[position].setOrigin(xyz);
m_start_transforms[position]
.setRotation(getArenaStartRotation(xyz, h));
}
else if (name == "camera")
{
node->get("far", &m_camera_far);
}
else if (name == "checks")
{
m_check_manager->load(*node);
}
else if (name == "particle-emitter")
{
if (UserConfigParams::m_particles_effects > 1)
{
m_track_object_manager->add(*node, parent, model_def_loader, parent_library);
}
}
else if (name == "sky-dome" || name == "sky-box" || name == "sky-color")
{
handleSky(*node, path);
}
else if (name == "end-cameras")
{
CameraEnd::readEndCamera(*node);
}
else if (name == "light")
{
m_track_object_manager->add(*node, parent, model_def_loader, parent_library);
}
else if (name == "weather")
{
std::string weather_particles;
node->get("particles", &weather_particles);
node->get("lightning", &m_weather_lightning);
node->get("sound", &m_weather_sound);
if (weather_particles.size() > 0)
{
m_sky_particles =
ParticleKindManager::get()->getParticles(weather_particles);
}
}
else if (name == "sun")
{
// handled above
}
else if (name == "lod")
{
// handled above
}
else if (name == "lightshaft")
{
// handled above
}
else if (name == "instancing")
{
// TODO: eventually remove, this is now automatic
}
else if (name == "subtitles")
{
std::vector<XMLNode*> subtitles;
node->getNodes("subtitle", subtitles);
for (unsigned int i = 0; i < subtitles.size(); i++)
{
int from = -1, to = -1;
std::string subtitle_text;
subtitles[i]->get("from", &from);
subtitles[i]->get("to", &to);
subtitles[i]->get("text", &subtitle_text);
if (from != -1 && to != -1 && subtitle_text.size() > 0)
{
m_subtitles.push_back( Subtitle(from, to, _(subtitle_text.c_str())) );
}
}
}
else
{
Log::warn("track", "While loading track '%s', element '%s' was "
"met but is unknown.",
m_ident.c_str(), node->getName().c_str());
}
} // for i<root->getNumNodes()
}
//-----------------------------------------------------------------------------
/** Handles a sky-dome or sky-box. It takes the xml node with the
* corresponding data for the sky and stores the corresponding data in
* the corresponding data structures.
* \param xml_node XML node with the sky data.
* \param filename Name of the file which is read, only used to print
* meaningful error messages.
*/
void Track::handleSky(const XMLNode &xml_node, const std::string &filename)
{
if(xml_node.getName()=="sky-box")
{
std::string s;
xml_node.get("texture", &s);
std::vector<std::string> v = StringUtils::split(s, ' ');
for (unsigned int i = 0; i<v.size(); i++)
{
void* obj = NULL;
#ifndef SERVER_ONLY
if (CVS->isGLSL())
{
video::IImage* img = getSkyTexture(v[i]);
obj = img;
}
else
#endif // !SERVER_ONLY
{
#ifndef SERVER_ONLY
if (GE::getDriver()->getDriverType() == video::EDT_VULKAN)
{
io::path p = file_manager->searchTexture(v[i]).c_str();
if (!p.empty())
{
io::path fullpath = file_manager->getFileSystem()
->getAbsolutePath(p).c_str();
GE::getGEConfig()->m_ondemand_load_texture_paths.
insert(fullpath.c_str());
}
}
#endif
video::ITexture* t = irr_driver->getTexture(v[i]);
if (t)
{
t->grab();
obj = t;
}
}
if (obj)
{
m_sky_textures.push_back(obj);
}
else
{
Log::error("track","Sky-box texture '%s' not found - ignored.",
v[i].c_str());
}
} // for i<v.size()
if(m_sky_textures.size()!=6)
{
Log::error("track",
"A skybox needs 6 textures, but %d are specified",
(int)m_sky_textures.size());
Log::error("track", "in '%s'.", filename.c_str());
}
else
{
m_sky_type = SKY_BOX;
}
std::string sh_textures;
xml_node.get("sh-texture", &sh_textures);
v = StringUtils::split(sh_textures, ' ');
#ifndef SERVER_ONLY
for (unsigned int i = 0; i<v.size(); i++)
{
if (CVS->isGLSL())
{
video::IImage* img = getSkyTexture(v[i]);
if (img)
{
m_spherical_harmonics_textures.push_back(img);
}
else
{
Log::error("track", "Sky-box spherical harmonics texture '%s' not found - ignored.",
v[i].c_str());
}
}
else
{
// We only need m_spherical_harmonics_textures.size()
m_spherical_harmonics_textures.push_back((void*)0x0);
}
} // for i<v.size()
#endif // !SERVER_ONLY
}
else if (xml_node.getName() == "sky-color")
{
m_sky_type = SKY_COLOR;
xml_node.get("rgb", &m_sky_color);
} // if sky-box
} // handleSky
//-----------------------------------------------------------------------------
Vec3 Track::flagCommand(const XMLNode *node)
{
Vec3 xyz;
// Set some kind of default in case Y is not defined in the file
// (with the new track exporter it always is defined anyway).
// Y is the height from which the item is dropped on the track.
xyz.setY(1000);
node->getXYZ(&xyz);
if (!m_track_mesh)
return xyz;
Vec3 loc(xyz);
// Test if the item lies on a 3d node, if so adjust the normal
// Also do a raycast if drop item is given
Vec3 normal(0, 1, 0);
Vec3 quad_normal = normal;
Vec3 hit_point = loc;
if (Graph::get())
{
int road_sector = Graph::UNKNOWN_SECTOR;
Graph::get()->findRoadSector(xyz, &road_sector);
// Only do custom direction of raycast if item is on quad graph
if (road_sector != Graph::UNKNOWN_SECTOR)
{
quad_normal = Graph::get()->getQuad(road_sector)->getNormal();
}
}
const Material *m;
// If raycast is used, increase the start position slightly
// in case that the point is too close to the actual surface
// (e.g. floating point errors can cause a problem here).
loc += quad_normal * 0.1f;
#ifndef DEBUG
m_track_mesh->castRay(loc, loc + (-10000 * quad_normal), &hit_point,
&m, &normal);
#else
bool drop_success = m_track_mesh->castRay(loc, loc +
(-10000 * quad_normal), &hit_point, &m, &normal);
if (!drop_success)
{
Log::warn("track", "flag at position (%f,%f,%f) can not be dropped",
loc.getX(), loc.getY(), loc.getZ());
Log::warn("track", "onto terrain - position unchanged.");
}
#endif
m_track_object_manager->castRay
(loc, loc + (-10000 * quad_normal), &hit_point, &m, &normal,
/*interpolate*/false);
const std::string &name = node->getName();
if (name == "red-flag")
{
m_red_flag = btTransform(shortestArcQuat(Vec3(0, 1, 0), normal),
hit_point);
}
else
{
m_blue_flag = btTransform(shortestArcQuat(Vec3(0, 1, 0), normal),
hit_point);
}
return hit_point;
} // flagCommand
//-----------------------------------------------------------------------------
/** Handle creation and placement of an item.
* \param xyz The position of the item.
* \param type The item type.
* \param drop True if the item Z position should be determined based on
* the track topology.
*/
void Track::itemCommand(const XMLNode *node)
{
const std::string &name = node->getName();
const bool is_mode_ctf = m_is_ctf && RaceManager::get()->isCTFMode();
bool ctf = false;
node->get("ctf", &ctf);
if ((is_mode_ctf && !ctf) || (!is_mode_ctf && ctf))
return;
Item::ItemType type;
if (name=="banana" ) type = Item::ITEM_BANANA;
else if(name=="item" ) type = Item::ITEM_BONUS_BOX;
else if(name=="small-nitro") type = Item::ITEM_NITRO_SMALL;
else if(name=="easter-egg" ) type = Item::ITEM_EASTER_EGG;
else type = Item::ITEM_NITRO_BIG;
Vec3 xyz;
// Set some kind of default in case Y is not defined in the file
// (with the new track exporter it always is defined anyway).
// Y is the height from which the item is dropped on the track.
xyz.setY(1000);
node->getXYZ(&xyz);
bool drop=true;
node->get("drop", &drop);
// Some modes (e.g. time trial) don't have any bonus boxes
if(type==Item::ITEM_BONUS_BOX &&
!World::getWorld()->haveBonusBoxes())
return;
// Only do easter eggs in easter egg mode.
if(!(RaceManager::get()->isEggHuntMode()) && type==Item::ITEM_EASTER_EGG)
{
Log::warn("track",
"Found easter egg in non-easter-egg mode - ignored.\n");
return;
}
Vec3 loc(xyz);
// Test if the item lies on a 3d node, if so adjust the normal
// Also do a raycast if drop item is given
Vec3 normal(0, 1, 0);
Vec3 quad_normal = normal;
Vec3 hit_point = loc;
if (Graph::get())
{
int road_sector = Graph::UNKNOWN_SECTOR;
Graph::get()->findRoadSector(xyz, &road_sector);
// Only do custom direction of raycast if item is on quad graph
if (road_sector != Graph::UNKNOWN_SECTOR)
{
quad_normal = Graph::get()->getQuad(road_sector)->getNormal();
}
}
if (drop)
{
const Material *m;
// If raycast is used, increase the start position slightly
// in case that the point is too close to the actual surface
// (e.g. floating point errors can cause a problem here).
loc += quad_normal * 0.1f;
#ifndef DEBUG
m_track_mesh->castRay(loc, loc + (-10000 * quad_normal), &hit_point,
&m, &normal);
m_track_object_manager->castRay(loc,
loc + (-10000 * quad_normal), &hit_point, &m, &normal,
/*interpolate*/false);
#else
bool drop_success = m_track_mesh->castRay(loc, loc +
(-10000 * quad_normal), &hit_point, &m, &normal);
bool over_driveable = m_track_object_manager->castRay(loc,
loc + (-10000 * quad_normal), &hit_point, &m, &normal,
/*interpolate*/false);
if (!drop_success && !over_driveable)
{
Log::warn("track",
"Item at position (%f,%f,%f) can not be dropped",
loc.getX(), loc.getY(), loc.getZ());
Log::warn("track", "onto terrain - position unchanged.");
}
#endif
}
m_item_manager->placeItem(type, drop ? hit_point : loc, normal);
} // itemCommand
// ----------------------------------------------------------------------------
std::vector< std::vector<float> > Track::buildHeightMap()
{
assert(m_height_map_mesh != NULL);
std::vector< std::vector<float> > out(HEIGHT_MAP_RESOLUTION);
float x = m_aabb_min.getX();
const float x_len = m_aabb_max.getX() - m_aabb_min.getX();
const float z_len = m_aabb_max.getZ() - m_aabb_min.getZ();
const float x_step = x_len/HEIGHT_MAP_RESOLUTION;
const float z_step = z_len/HEIGHT_MAP_RESOLUTION;
btVector3 hitpoint;
const Material* material;
btVector3 normal;
for (int i=0; i<HEIGHT_MAP_RESOLUTION; i++)
{
out[i].resize(HEIGHT_MAP_RESOLUTION);
float z = m_aabb_min.getZ();
for (int j=0; j<HEIGHT_MAP_RESOLUTION; j++)
{
btVector3 pos(x, 100.0f, z);
btVector3 to = pos;
to.setY(-100000.f);
m_height_map_mesh->castRay(pos, to, &hitpoint, &material, &normal);
z += z_step;
out[i][j] = hitpoint.getY();
} // j<HEIGHT_MAP_RESOLUTION
x += x_step;
}
return out;
} // buildHeightMap
// ----------------------------------------------------------------------------
void Track::drawMiniMap(const core::rect<s32>& dest_rect) const
{
if(m_render_target)
m_render_target->draw2DImage(dest_rect, NULL,
video::SColor(127, 255, 255, 255),
true);
}
// ----------------------------------------------------------------------------
/** Returns the rotation of the sun. */
const core::vector3df& Track::getSunRotation()
{
return m_sun->getRotation();
}
//-----------------------------------------------------------------------------
bool Track::isOnGround(const Vec3& xyz, const Vec3& down, Vec3* hit_point,
Vec3* normal, bool print_warning)
{
// Material and hit point are not needed;
const Material *m;
bool over_ground = m_track_mesh->castRay(xyz, down, hit_point,
&m, normal);
// Now also raycast against all track objects (that are driveable). If
// there should be a closer result (than the one against the main track
// mesh), its data will be returned.
// From TerrainInfo::update
bool over_driveable = m_track_object_manager->castRay(xyz, down,
hit_point, &m, normal, /*interpolate*/false);
if (!over_ground && !over_driveable)
{
if (print_warning)
{
Log::warn("physics", "Kart at (%f %f %f) can not be dropped.",
xyz.getX(),xyz.getY(),xyz.getZ());
}
return false;
}
// Check if the material the kart is about to be placed on would trigger
// a reset. If so, this is not a valid position.
if(m && m->isDriveReset())
{
if (print_warning)
{
Log::warn("physics","Kart at (%f %f %f) over reset terrain '%s'",
xyz.getX(),xyz.getY(),xyz.getZ(),
m->getTexFname().c_str());
}
return false;
}
// See if the kart is too high above the ground - it would drop
// too long.
if(xyz.getY() - hit_point->getY() > 5)
{
if (print_warning)
{
Log::warn("physics",
"Kart at (%f %f %f) is too high above ground at (%f %f %f)",
xyz.getX(),xyz.getY(),xyz.getZ(),
hit_point->getX(),hit_point->getY(),hit_point->getZ());
}
return false;
}
return true;
} // isOnGround
//-----------------------------------------------------------------------------
/** Determines if the kart is over ground.
* Used in setting the starting positions of all the karts.
* \param k The kart to project downward.
* \return True of the kart is on terrain.
*/
bool Track::findGround(AbstractKart *kart)
{
const Vec3 &xyz = kart->getXYZ();
Vec3 down = kart->getTrans().getBasis() * Vec3(0, -10000.0f, 0);
Vec3 hit_point, normal;
if (!isOnGround(xyz, down, &hit_point, &normal))
return false;
btTransform t = kart->getBody()->getCenterOfMassTransform();
// The computer offset is slightly too large, it should take
// the default suspension rest instead of suspension rest (i.e. the
// length of the suspension with the weight of the kart resting on
// it). On the other hand this initial bouncing looks nice imho
// - so I'll leave it in for now.
float offset = kart->getKartProperties()->getSuspensionRest();
t.setOrigin(hit_point + normal * offset);
kart->getBody()->setCenterOfMassTransform(t);
kart->setTrans(t);
return true;
} // findGround
//-----------------------------------------------------------------------------
float Track::getTrackLength() const
{
return DriveGraph::get()->getLapLength();
} // getTrackLength
//-----------------------------------------------------------------------------
float Track::getAngle(int n) const
{
return DriveGraph::get()->getAngleToNext(n, 0);
} // getAngle
//-----------------------------------------------------------------------------
void Track::uploadNodeVertexBuffer(scene::ISceneNode *node)
{
#ifndef SERVER_ONLY
if (!CVS->isGLSL())
{
return;
}
SP::SPMeshNode* spmn = dynamic_cast<SP::SPMeshNode*>(node);
if (spmn)
{
SP::uploadSPM(spmn->getSPM());
}
#endif
} // uploadNodeVertexBuffer
//-----------------------------------------------------------------------------
void Track::copyFromMainProcess()
{
// Clear all unneeded objects copied in main process track
m_physical_object_uid = 0;
m_animated_textures.clear();
m_animated_textures.shrink_to_fit();
m_all_nodes.clear();
m_all_nodes.shrink_to_fit();
m_static_physics_only_nodes.clear();
m_static_physics_only_nodes.shrink_to_fit();
m_object_physics_only_nodes.clear();
m_object_physics_only_nodes.shrink_to_fit();
m_sun = NULL;
m_all_cached_meshes.clear();
m_all_cached_meshes.shrink_to_fit();
m_detached_cached_meshes.clear();
m_detached_cached_meshes.shrink_to_fit();
m_sky_textures.clear();
m_sky_textures.shrink_to_fit();
m_spherical_harmonics_textures.clear();
m_spherical_harmonics_textures.shrink_to_fit();
m_meta_library.clear();
m_meta_library.shrink_to_fit();
// Clone the needed object now in main process
Track* main_track = m_current_track[PT_MAIN];
CheckManager* main_cm = main_track->m_check_manager;
m_check_manager = new CheckManager();
for (unsigned i = 0; i < main_cm->getCheckStructureCount(); i++)
{
CheckStructure* cs = main_cm->getCheckStructure(i);
m_check_manager->add(cs->clone());
}
TrackObjectManager* main_tom = m_track_object_manager;
m_track_object_manager = new TrackObjectManager();
for (auto* to : main_tom->getObjects().m_contents_vector)
{
TrackObject* clone = to->cloneToChild();
if (clone)
{
m_track_object_manager->insertObject(clone);
m_track_object_manager->insertDriveableObject(clone);
}
}
m_track_mesh = new TriangleMesh(/*can_be_transformed*/false);
m_height_map_mesh = NULL;
m_gfx_effect_mesh = new TriangleMesh(/*can_be_transformed*/false);
m_track_mesh->copyFrom(*main_track->m_track_mesh);
m_gfx_effect_mesh->copyFrom(*main_track->m_gfx_effect_mesh);
// At the moment we only use network for child track
auto nim = std::make_shared<NetworkItemManager>();
for (unsigned i = 0; i < m_item_manager->getNumberOfItems(); i++)
{
ItemState* it = m_item_manager->getItem(i);
nim->insertItem(new Item(it->getType(), it->getXYZ(), it->getNormal(),
NULL/*mesh*/, NULL/*lowres_mesh*/, "", NULL/*owner*/));
}
m_item_manager = nim;
} // copyFromMainProcess
//-----------------------------------------------------------------------------
void Track::initChildTrack()
{
// This will be called in child process after main one copied to it
assert(STKProcess::getType() == PT_CHILD);
// Add in child process for rewind manager
std::dynamic_pointer_cast<NetworkItemManager>
(m_item_manager)->rewinderAdd();
std::dynamic_pointer_cast<NetworkItemManager>
(m_item_manager)->initServer();
// We call physics init in child process too
Physics::get()->init(m_aabb_min, m_aabb_max);
m_track_mesh->createPhysicalBody(m_friction);
m_gfx_effect_mesh->createCollisionShape();
// All child track objects are only cloned if they have physical objects
for (auto* to : m_track_object_manager->getObjects().m_contents_vector)
to->getPhysicalObject()->addBody();
m_track_object_manager->init();
if (auto sl = LobbyProtocol::get<ServerLobby>())
{
sl->saveInitialItems(
std::dynamic_pointer_cast<NetworkItemManager>(m_item_manager));
}
} // initChildTrack
//-----------------------------------------------------------------------------
void Track::cleanChildTrack()
{
assert(STKProcess::getType() == PT_CHILD);
Track* child_track = m_current_track[PT_CHILD];
child_track->m_item_manager = nullptr;
delete child_track->m_check_manager;
delete child_track->m_track_object_manager;
delete child_track->m_track_mesh;
delete child_track->m_gfx_effect_mesh;
delete child_track;
m_current_track[PT_CHILD] = NULL;
} // cleanChildTrack
//-----------------------------------------------------------------------------
video::IImage* Track::getSkyTexture(std::string path) const
{
#ifdef SERVER_ONLY
return NULL;
#else
if (path.find('/') == std::string::npos)
{
io::path relative_path = file_manager->searchTexture(path).c_str();
if (relative_path.empty())
return NULL;
path = file_manager->getFileSystem()->getAbsolutePath(relative_path)
.c_str();
}
return GE::getResizedImage(path, irr_driver->getVideoDriver()
->getDriverAttributes().getAttributeAsDimension2d("MAX_TEXTURE_SIZE"));
#endif
} // getSkyTexture