Fix transparency sorting and animation faraway check not using mesh chunk bounding sphere

This commit is contained in:
Desour 2024-08-24 11:31:36 +02:00 committed by sfan5
parent 2e883189c1
commit 0f7ee126de

@ -767,15 +767,12 @@ void ClientMap::renderMap(video::IVideoDriver* driver, s32 pass)
if (is_frustum_culled(mesh_sphere_center, mesh_sphere_radius)) if (is_frustum_culled(mesh_sphere_center, mesh_sphere_radius))
continue; continue;
v3f block_pos_r = intToFloat(block->getPosRelative() + MAP_BLOCKSIZE / 2, BS);
float d = camera_position.getDistanceFrom(block_pos_r);
d = MYMAX(0,d - BLOCK_MAX_RADIUS);
// Mesh animation // Mesh animation
if (pass == scene::ESNRP_SOLID) { if (pass == scene::ESNRP_SOLID) {
// Pretty random but this should work somewhat nicely // 50 nodes is pretty arbitrary but it should work somewhat nicely
bool faraway = d >= BS * 50; float distance_sq = camera_position.getDistanceFromSQ(mesh_sphere_center);
bool faraway = distance_sq >= std::pow(BS * 50 + mesh_sphere_radius, 2.0f);
if (block_mesh->isAnimationForced() || !faraway || if (block_mesh->isAnimationForced() || !faraway ||
mesh_animate_count < (m_control.range_all ? 200 : 50)) { mesh_animate_count < (m_control.range_all ? 200 : 50)) {
@ -1305,27 +1302,29 @@ void ClientMap::updateTransparentMeshBuffers()
ScopeProfiler sp(g_profiler, "CM::updateTransparentMeshBuffers", SPT_AVG); ScopeProfiler sp(g_profiler, "CM::updateTransparentMeshBuffers", SPT_AVG);
u32 sorted_blocks = 0; u32 sorted_blocks = 0;
u32 unsorted_blocks = 0; u32 unsorted_blocks = 0;
f32 sorting_distance_sq = std::pow(m_cache_transparency_sorting_distance * BS, 2.0f); f32 sorting_distance = m_cache_transparency_sorting_distance * BS;
// Update the order of transparent mesh buffers in each mesh // Update the order of transparent mesh buffers in each mesh
for (auto it = m_drawlist.begin(); it != m_drawlist.end(); it++) { for (auto it = m_drawlist.begin(); it != m_drawlist.end(); it++) {
MapBlock* block = it->second; MapBlock *block = it->second;
if (!block->mesh) MapBlockMesh *blockmesh = block->mesh;
if (!blockmesh)
continue; continue;
if (m_needs_update_transparent_meshes || if (m_needs_update_transparent_meshes ||
block->mesh->getTransparentBuffers().size() == 0) { blockmesh->getTransparentBuffers().size() == 0) {
v3s16 block_pos = block->getPos(); v3f mesh_sphere_center = intToFloat(block->getPosRelative(), BS)
v3f block_pos_f = intToFloat(block_pos * MAP_BLOCKSIZE + MAP_BLOCKSIZE / 2, BS); + blockmesh->getBoundingSphereCenter();
f32 distance = m_camera_position.getDistanceFromSQ(block_pos_f); f32 mesh_sphere_radius = blockmesh->getBoundingRadius();
if (distance <= sorting_distance_sq) { f32 distance_sq = m_camera_position.getDistanceFromSQ(mesh_sphere_center);
block->mesh->updateTransparentBuffers(m_camera_position, block_pos);
if (distance_sq <= std::pow(sorting_distance + mesh_sphere_radius, 2.0f)) {
blockmesh->updateTransparentBuffers(m_camera_position, block->getPos());
++sorted_blocks; ++sorted_blocks;
} } else {
else { blockmesh->consolidateTransparentBuffers();
block->mesh->consolidateTransparentBuffers();
++unsorted_blocks; ++unsorted_blocks;
} }
} }