Fix minimap bug

This commit is contained in:
Aidar Shaikhiev 2024-07-28 20:59:47 +00:00 committed by Marcel
parent 0e64ac29ad
commit 88d85108d5
2 changed files with 29 additions and 4 deletions

View File

@ -52,6 +52,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Limit the minimum vertical scaling value in the world generation UI to 0.1 to prevent an overflow - Limit the minimum vertical scaling value in the world generation UI to 0.1 to prevent an overflow
- Wood log and worker/linen clothing pricing. - Wood log and worker/linen clothing pricing.
- Charm recipes can now be found in the Potions tab of the crafting menu - Charm recipes can now be found in the Potions tab of the crafting menu
- Black minimap when some terrain blocks are deleted.
## [0.16.0] - 2024-03-30 ## [0.16.0] - 2024-03-30

View File

@ -21,12 +21,13 @@ use common::{
vol::{ReadVol, RectVolSize}, vol::{ReadVol, RectVolSize},
}; };
use common_net::msg::world_msg::SiteKind; use common_net::msg::world_msg::SiteKind;
use common_state::TerrainChanges;
use conrod_core::{ use conrod_core::{
color, position, color, position,
widget::{self, Button, Image, Rectangle, Text}, widget::{self, Button, Image, Rectangle, Text},
widget_ids, Color, Colorable, Positionable, Sizeable, Widget, WidgetCommon, widget_ids, Color, Colorable, Positionable, Sizeable, Widget, WidgetCommon,
}; };
use hashbrown::HashMap; use hashbrown::{HashMap, HashSet};
use image::{DynamicImage, RgbaImage}; use image::{DynamicImage, RgbaImage};
use specs::WorldExt; use specs::WorldExt;
use std::sync::Arc; use std::sync::Arc;
@ -45,6 +46,7 @@ struct MinimapColumn {
pub struct VoxelMinimap { pub struct VoxelMinimap {
chunk_minimaps: HashMap<Vec2<i32>, MinimapColumn>, chunk_minimaps: HashMap<Vec2<i32>, MinimapColumn>,
chunks_to_replace: HashSet<Vec2<i32>>,
composited: RgbaImage, composited: RgbaImage,
image_id: img_ids::Rotations, image_id: img_ids::Rotations,
last_pos: Vec3<i32>, last_pos: Vec3<i32>,
@ -63,6 +65,7 @@ impl VoxelMinimap {
); );
Self { Self {
chunk_minimaps: HashMap::new(), chunk_minimaps: HashMap::new(),
chunks_to_replace: HashSet::new(),
image_id: ui.add_graphic_with_rotations(Graphic::Image( image_id: ui.add_graphic_with_rotations(Graphic::Image(
Arc::new(DynamicImage::ImageRgba8(composited.clone())), Arc::new(DynamicImage::ImageRgba8(composited.clone())),
Some(Rgba::from([0.0, 0.0, 0.0, 0.0])), Some(Rgba::from([0.0, 0.0, 0.0, 0.0])),
@ -162,7 +165,8 @@ impl VoxelMinimap {
let delta: Vec2<u32> = (key - cpos).map(i32::abs).as_(); let delta: Vec2<u32> = (key - cpos).map(i32::abs).as_();
if delta.x < VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.x if delta.x < VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.x
&& delta.y < VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.y && delta.y < VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.y
&& !self.chunk_minimaps.contains_key(&key) && (!self.chunk_minimaps.contains_key(&key)
|| self.chunks_to_replace.contains(&key))
{ {
if let Some((_, column)) = self.keyed_jobs.spawn(Some(pool), key, || { if let Some((_, column)) = self.keyed_jobs.spawn(Some(pool), key, || {
let arc_chunk = Arc::clone(chunk); let arc_chunk = Arc::clone(chunk);
@ -198,6 +202,7 @@ impl VoxelMinimap {
} }
} }
}) { }) {
self.chunks_to_replace.remove(&key);
self.chunk_minimaps.insert(key, column); self.chunk_minimaps.insert(key, column);
new_chunks = true; new_chunks = true;
} }
@ -206,13 +211,30 @@ impl VoxelMinimap {
new_chunks new_chunks
} }
fn add_chunks_to_replace(&mut self, terrain: &TerrainGrid, changes: &TerrainChanges) {
changes
.modified_blocks
.iter()
.filter(|(key, old_block)| {
terrain.get(**key).map_or(false, |new_block| {
new_block.is_terrain() != old_block.is_terrain()
})
})
.map(|(key, _)| terrain.pos_key(*key))
.for_each(|key| {
self.chunks_to_replace.insert(key);
});
}
fn remove_chunks_far(&mut self, terrain: &TerrainGrid, cpos: Vec2<i32>) { fn remove_chunks_far(&mut self, terrain: &TerrainGrid, cpos: Vec2<i32>) {
self.chunk_minimaps.retain(|key, _| { let key_predicate = |key: &Vec2<i32>| {
let delta: Vec2<u32> = (key - cpos).map(i32::abs).as_(); let delta: Vec2<u32> = (key - cpos).map(i32::abs).as_();
delta.x < 1 + VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.x delta.x < 1 + VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.x
&& delta.y < 1 + VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.y && delta.y < 1 + VOXEL_MINIMAP_SIDELENGTH / TerrainChunkSize::RECT_SIZE.y
&& terrain.get_key(*key).is_some() && terrain.get_key(*key).is_some()
}); };
self.chunks_to_replace.retain(&key_predicate);
self.chunk_minimaps.retain(|key, _| key_predicate(key));
} }
pub fn maintain(&mut self, client: &Client, ui: &mut Ui) { pub fn maintain(&mut self, client: &Client, ui: &mut Ui) {
@ -229,6 +251,8 @@ impl VoxelMinimap {
let pool = client.state().ecs().read_resource::<SlowJobPool>(); let pool = client.state().ecs().read_resource::<SlowJobPool>();
let terrain = client.state().terrain(); let terrain = client.state().terrain();
let changed_blocks = client.state().terrain_changes();
self.add_chunks_to_replace(&terrain, &changed_blocks);
let new_chunks = self.add_chunks_near(&pool, &terrain, cpos); let new_chunks = self.add_chunks_near(&pool, &terrain, cpos);
self.remove_chunks_far(&terrain, cpos); self.remove_chunks_far(&terrain, cpos);