use crate::{Rect, TextureAtlas}; use bevy_asset::{Assets, Handle}; use bevy_math::Vec2; use bevy_render::texture::{Extent3d, Texture, TextureDimension, TextureFormat}; use bevy_utils::HashMap; use rectangle_pack::{ contains_smallest_box, pack_rects, volume_heuristic, GroupedRectsToPlace, PackedLocation, RectToInsert, TargetBin, }; use thiserror::Error; #[derive(Debug)] pub struct TextureAtlasBuilder { pub textures: Vec>, pub rects_to_place: GroupedRectsToPlace>, pub initial_size: Vec2, pub max_size: Vec2, } impl Default for TextureAtlasBuilder { fn default() -> Self { Self::new(Vec2::new(256., 256.), Vec2::new(2048., 2048.)) } } #[derive(Debug, Error)] pub enum RectanglePackError { #[error("Could not pack textures into an atlas within the given bounds")] NotEnoughSpace, } impl TextureAtlasBuilder { pub fn new(initial_size: Vec2, max_size: Vec2) -> Self { Self { textures: Default::default(), rects_to_place: GroupedRectsToPlace::new(), initial_size, max_size, } } pub fn add_texture(&mut self, texture_handle: Handle, texture: &Texture) { self.rects_to_place.push_rect( texture_handle, None, RectToInsert::new(texture.size.width, texture.size.height, 1), ) } fn place_texture( &mut self, atlas_texture: &mut Texture, texture: &Texture, packed_location: &PackedLocation, ) { let rect_width = packed_location.width() as usize; let rect_height = packed_location.height() as usize; let rect_x = packed_location.x() as usize; let rect_y = packed_location.y() as usize; let atlas_width = atlas_texture.size.width as usize; let format_size = atlas_texture.format.pixel_size(); for (texture_y, bound_y) in (rect_y..rect_y + rect_height).enumerate() { let begin = (bound_y * atlas_width + rect_x) * format_size; let end = begin + rect_width * format_size; let texture_begin = texture_y * rect_width * format_size; let texture_end = texture_begin + rect_width * format_size; atlas_texture.data[begin..end] .copy_from_slice(&texture.data[texture_begin..texture_end]); } } pub fn finish( mut self, textures: &mut Assets, ) -> Result { let initial_width = self.initial_size.x as u32; let initial_height = self.initial_size.y as u32; let max_width = self.max_size.x as u32; let max_height = self.max_size.y as u32; let mut current_width = initial_width; let mut current_height = initial_height; let mut rect_placements = None; let mut atlas_texture = Texture::default(); while rect_placements.is_none() { if current_width > max_width || current_height > max_height { break; } let last_attempt = current_height == max_height && current_width == max_width; let mut target_bins = std::collections::BTreeMap::new(); target_bins.insert(0, TargetBin::new(current_width, current_height, 1)); rect_placements = match pack_rects( &self.rects_to_place, target_bins, &volume_heuristic, &contains_smallest_box, ) { Ok(rect_placements) => { atlas_texture = Texture::new_fill( Extent3d::new(current_width, current_height, 1), TextureDimension::D2, &[0, 0, 0, 0], TextureFormat::Rgba8UnormSrgb, ); Some(rect_placements) } Err(rectangle_pack::RectanglePackError::NotEnoughBinSpace) => { current_height = bevy_math::clamp(current_height * 2, 0, max_height); current_width = bevy_math::clamp(current_width * 2, 0, max_width); None } }; if last_attempt { break; } } let rect_placements = rect_placements.ok_or(RectanglePackError::NotEnoughSpace)?; let mut texture_rects = Vec::with_capacity(rect_placements.packed_locations().len()); let mut texture_handles = HashMap::default(); for (texture_handle, (_, packed_location)) in rect_placements.packed_locations().iter() { let texture = textures.get(texture_handle).unwrap(); let min = Vec2::new(packed_location.x() as f32, packed_location.y() as f32); let max = min + Vec2::new( packed_location.width() as f32, packed_location.height() as f32, ); texture_handles.insert(texture_handle.clone_weak(), texture_rects.len()); texture_rects.push(Rect { min, max }); self.place_texture(&mut atlas_texture, texture, packed_location); } Ok(TextureAtlas { size: atlas_texture.size.as_vec3().truncate(), texture: textures.add(atlas_texture), textures: texture_rects, texture_handles: Some(texture_handles), }) } }