diff options
Diffstat (limited to '')
| -rw-r--r-- | graphics/src/image/raster.rs (renamed from wgpu/src/image/raster.rs) | 86 | ||||
| -rw-r--r-- | graphics/src/image/vector.rs (renamed from wgpu/src/image/vector.rs) | 60 | ||||
| -rw-r--r-- | wgpu/src/image.rs | 68 | ||||
| -rw-r--r-- | wgpu/src/image/atlas.rs | 205 | ||||
| -rw-r--r-- | wgpu/src/image/atlas/allocation.rs | 6 | ||||
| -rw-r--r-- | wgpu/src/image/atlas/allocator.rs | 4 | ||||
| -rw-r--r-- | wgpu/src/image/atlas/entry.rs | 10 | 
7 files changed, 245 insertions, 194 deletions
| diff --git a/wgpu/src/image/raster.rs b/graphics/src/image/raster.rs index 2b4d4af3..da46c30f 100644 --- a/wgpu/src/image/raster.rs +++ b/graphics/src/image/raster.rs @@ -1,43 +1,53 @@ -use crate::image::atlas::{self, Atlas}; +//! Raster image loading and caching. +use crate::image::Storage; +use crate::Size; +  use iced_native::image; -use std::collections::{HashMap, HashSet};  use bitflags::bitflags; +use std::collections::{HashMap, HashSet}; +/// Entry in cache corresponding to an image handle  #[derive(Debug)] -pub enum Memory { -    Host(::image_rs::ImageBuffer<::image_rs::Bgra<u8>, Vec<u8>>), -    Device(atlas::Entry), +pub enum Memory<T: Storage> { +    /// Image data on host +    Host(::image_rs::ImageBuffer<::image_rs::Rgba<u8>, Vec<u8>>), +    /// Storage entry +    Device(T::Entry), +    /// Image not found      NotFound, +    /// Invalid image data      Invalid,  } -impl Memory { -    pub fn dimensions(&self) -> (u32, u32) { +impl<T: Storage> Memory<T> { +    /// Width and height of image +    pub fn dimensions(&self) -> Size<u32> { +        use crate::image::storage::Entry; +          match self { -            Memory::Host(image) => image.dimensions(), +            Memory::Host(image) => { +                let (width, height) = image.dimensions(); + +                Size::new(width, height) +            }              Memory::Device(entry) => entry.size(), -            Memory::NotFound => (1, 1), -            Memory::Invalid => (1, 1), +            Memory::NotFound => Size::new(1, 1), +            Memory::Invalid => Size::new(1, 1),          }      }  } +/// Caches image raster data  #[derive(Debug)] -pub struct Cache { -    map: HashMap<u64, Memory>, +pub struct Cache<T: Storage> { +    map: HashMap<u64, Memory<T>>,      hits: HashSet<u64>,  } -impl Cache { -    pub fn new() -> Self { -        Self { -            map: HashMap::new(), -            hits: HashSet::new(), -        } -    } - -    pub fn load(&mut self, handle: &image::Handle) -> &mut Memory { +impl<T: Storage> Cache<T> { +    /// Load image +    pub fn load(&mut self, handle: &image::Handle) -> &mut Memory<T> {          if self.contains(handle) {              return self.get(handle).unwrap();          } @@ -53,7 +63,7 @@ impl Cache {                          })                          .unwrap_or_else(Operation::empty); -                    Memory::Host(operation.perform(image.to_bgra8())) +                    Memory::Host(operation.perform(image.to_rgba8()))                  } else {                      Memory::NotFound                  } @@ -65,12 +75,12 @@ impl Cache {                              .ok()                              .unwrap_or_else(Operation::empty); -                    Memory::Host(operation.perform(image.to_bgra8())) +                    Memory::Host(operation.perform(image.to_rgba8()))                  } else {                      Memory::Invalid                  }              } -            image::Data::Pixels { +            image::Data::Rgba {                  width,                  height,                  pixels, @@ -91,19 +101,19 @@ impl Cache {          self.get(handle).unwrap()      } +    /// Load image and upload raster data      pub fn upload(          &mut self,          handle: &image::Handle, -        device: &wgpu::Device, -        encoder: &mut wgpu::CommandEncoder, -        atlas: &mut Atlas, -    ) -> Option<&atlas::Entry> { +        state: &mut T::State<'_>, +        storage: &mut T, +    ) -> Option<&T::Entry> {          let memory = self.load(handle);          if let Memory::Host(image) = memory {              let (width, height) = image.dimensions(); -            let entry = atlas.upload(width, height, image, device, encoder)?; +            let entry = storage.upload(width, height, image, state)?;              *memory = Memory::Device(entry);          } @@ -115,7 +125,8 @@ impl Cache {          }      } -    pub fn trim(&mut self, atlas: &mut Atlas) { +    /// Trim cache misses from cache +    pub fn trim(&mut self, storage: &mut T, state: &mut T::State<'_>) {          let hits = &self.hits;          self.map.retain(|k, memory| { @@ -123,7 +134,7 @@ impl Cache {              if !retain {                  if let Memory::Device(entry) = memory { -                    atlas.remove(entry); +                    storage.remove(entry, state);                  }              } @@ -133,13 +144,13 @@ impl Cache {          self.hits.clear();      } -    fn get(&mut self, handle: &image::Handle) -> Option<&mut Memory> { +    fn get(&mut self, handle: &image::Handle) -> Option<&mut Memory<T>> {          let _ = self.hits.insert(handle.id());          self.map.get_mut(&handle.id())      } -    fn insert(&mut self, handle: &image::Handle, memory: Memory) { +    fn insert(&mut self, handle: &image::Handle, memory: Memory<T>) {          let _ = self.map.insert(handle.id(), memory);      } @@ -148,6 +159,15 @@ impl Cache {      }  } +impl<T: Storage> Default for Cache<T> { +    fn default() -> Self { +        Self { +            map: HashMap::new(), +            hits: HashSet::new(), +        } +    } +} +  bitflags! {      struct Operation: u8 {          const FLIP_HORIZONTALLY = 0b001; diff --git a/wgpu/src/image/vector.rs b/graphics/src/image/vector.rs index b08a0aa2..dc271c9e 100644 --- a/wgpu/src/image/vector.rs +++ b/graphics/src/image/vector.rs @@ -1,46 +1,45 @@ -use crate::image::atlas::{self, Atlas}; +//! Vector image loading and caching +use crate::image::Storage;  use iced_native::svg; +use iced_native::Size;  use std::collections::{HashMap, HashSet};  use std::fs; +/// Entry in cache corresponding to an svg handle  pub enum Svg { +    /// Parsed svg      Loaded(usvg::Tree), +    /// Svg not found or failed to parse      NotFound,  }  impl Svg { -    pub fn viewport_dimensions(&self) -> (u32, u32) { +    /// Viewport width and height +    pub fn viewport_dimensions(&self) -> Size<u32> {          match self {              Svg::Loaded(tree) => {                  let size = tree.svg_node().size; -                (size.width() as u32, size.height() as u32) +                Size::new(size.width() as u32, size.height() as u32)              } -            Svg::NotFound => (1, 1), +            Svg::NotFound => Size::new(1, 1),          }      }  } +/// Caches svg vector and raster data  #[derive(Debug)] -pub struct Cache { +pub struct Cache<T: Storage> {      svgs: HashMap<u64, Svg>, -    rasterized: HashMap<(u64, u32, u32), atlas::Entry>, +    rasterized: HashMap<(u64, u32, u32), T::Entry>,      svg_hits: HashSet<u64>,      rasterized_hits: HashSet<(u64, u32, u32)>,  } -impl Cache { -    pub fn new() -> Self { -        Self { -            svgs: HashMap::new(), -            rasterized: HashMap::new(), -            svg_hits: HashSet::new(), -            rasterized_hits: HashSet::new(), -        } -    } - +impl<T: Storage> Cache<T> { +    /// Load svg      pub fn load(&mut self, handle: &svg::Handle) -> &Svg {          if self.svgs.contains_key(&handle.id()) {              return self.svgs.get(&handle.id()).unwrap(); @@ -73,15 +72,15 @@ impl Cache {          self.svgs.get(&handle.id()).unwrap()      } +    /// Load svg and upload raster data      pub fn upload(          &mut self,          handle: &svg::Handle,          [width, height]: [f32; 2],          scale: f32, -        device: &wgpu::Device, -        encoder: &mut wgpu::CommandEncoder, -        texture_atlas: &mut Atlas, -    ) -> Option<&atlas::Entry> { +        state: &mut T::State<'_>, +        storage: &mut T, +    ) -> Option<&T::Entry> {          let id = handle.id();          let (width, height) = ( @@ -125,12 +124,11 @@ impl Cache {                  let mut rgba = img.take();                  rgba.chunks_exact_mut(4).for_each(|rgba| rgba.swap(0, 2)); -                let allocation = texture_atlas.upload( +                let allocation = storage.upload(                      width,                      height,                      bytemuck::cast_slice(rgba.as_slice()), -                    device, -                    encoder, +                    state,                  )?;                  log::debug!("allocating {} {}x{}", id, width, height); @@ -144,7 +142,8 @@ impl Cache {          }      } -    pub fn trim(&mut self, atlas: &mut Atlas) { +    /// Load svg and upload raster data +    pub fn trim(&mut self, storage: &mut T, state: &mut T::State<'_>) {          let svg_hits = &self.svg_hits;          let rasterized_hits = &self.rasterized_hits; @@ -153,7 +152,7 @@ impl Cache {              let retain = rasterized_hits.contains(k);              if !retain { -                atlas.remove(entry); +                storage.remove(entry, state);              }              retain @@ -163,6 +162,17 @@ impl Cache {      }  } +impl<T: Storage> Default for Cache<T> { +    fn default() -> Self { +        Self { +            svgs: HashMap::new(), +            rasterized: HashMap::new(), +            svg_hits: HashSet::new(), +            rasterized_hits: HashSet::new(), +        } +    } +} +  impl std::fmt::Debug for Svg {      fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {          match self { diff --git a/wgpu/src/image.rs b/wgpu/src/image.rs index d964aed7..d06815bb 100644 --- a/wgpu/src/image.rs +++ b/wgpu/src/image.rs @@ -1,22 +1,23 @@  mod atlas; -#[cfg(feature = "image_rs")] -mod raster; +#[cfg(feature = "image")] +use iced_graphics::image::raster;  #[cfg(feature = "svg")] -mod vector; +use iced_graphics::image::vector;  use crate::Transformation;  use atlas::Atlas;  use iced_graphics::layer; -use iced_native::Rectangle; +use iced_native::{Rectangle, Size}; +  use std::cell::RefCell;  use std::mem;  use bytemuck::{Pod, Zeroable}; -#[cfg(feature = "image_rs")] +#[cfg(feature = "image")]  use iced_native::image;  #[cfg(feature = "svg")] @@ -24,10 +25,10 @@ use iced_native::svg;  #[derive(Debug)]  pub struct Pipeline { -    #[cfg(feature = "image_rs")] -    raster_cache: RefCell<raster::Cache>, +    #[cfg(feature = "image")] +    raster_cache: RefCell<raster::Cache<Atlas>>,      #[cfg(feature = "svg")] -    vector_cache: RefCell<vector::Cache>, +    vector_cache: RefCell<vector::Cache<Atlas>>,      pipeline: wgpu::RenderPipeline,      uniforms: wgpu::Buffer, @@ -242,11 +243,11 @@ impl Pipeline {          });          Pipeline { -            #[cfg(feature = "image_rs")] -            raster_cache: RefCell::new(raster::Cache::new()), +            #[cfg(feature = "image")] +            raster_cache: RefCell::new(raster::Cache::default()),              #[cfg(feature = "svg")] -            vector_cache: RefCell::new(vector::Cache::new()), +            vector_cache: RefCell::new(vector::Cache::default()),              pipeline,              uniforms: uniforms_buffer, @@ -261,8 +262,8 @@ impl Pipeline {          }      } -    #[cfg(feature = "image_rs")] -    pub fn dimensions(&self, handle: &image::Handle) -> (u32, u32) { +    #[cfg(feature = "image")] +    pub fn dimensions(&self, handle: &image::Handle) -> Size<u32> {          let mut cache = self.raster_cache.borrow_mut();          let memory = cache.load(handle); @@ -270,7 +271,7 @@ impl Pipeline {      }      #[cfg(feature = "svg")] -    pub fn viewport_dimensions(&self, handle: &svg::Handle) -> (u32, u32) { +    pub fn viewport_dimensions(&self, handle: &svg::Handle) -> Size<u32> {          let mut cache = self.vector_cache.borrow_mut();          let svg = cache.load(handle); @@ -290,7 +291,7 @@ impl Pipeline {      ) {          let instances: &mut Vec<Instance> = &mut Vec::new(); -        #[cfg(feature = "image_rs")] +        #[cfg(feature = "image")]          let mut raster_cache = self.raster_cache.borrow_mut();          #[cfg(feature = "svg")] @@ -298,12 +299,11 @@ impl Pipeline {          for image in images {              match &image { -                #[cfg(feature = "image_rs")] +                #[cfg(feature = "image")]                  layer::Image::Raster { handle, bounds } => {                      if let Some(atlas_entry) = raster_cache.upload(                          handle, -                        device, -                        encoder, +                        &mut (device, encoder),                          &mut self.texture_atlas,                      ) {                          add_instances( @@ -314,7 +314,7 @@ impl Pipeline {                          );                      }                  } -                #[cfg(not(feature = "image_rs"))] +                #[cfg(not(feature = "image"))]                  layer::Image::Raster { .. } => {}                  #[cfg(feature = "svg")] @@ -325,8 +325,7 @@ impl Pipeline {                          handle,                          size,                          _scale, -                        device, -                        encoder, +                        &mut (device, encoder),                          &mut self.texture_atlas,                      ) {                          add_instances( @@ -446,12 +445,20 @@ impl Pipeline {          }      } -    pub fn trim_cache(&mut self) { -        #[cfg(feature = "image_rs")] -        self.raster_cache.borrow_mut().trim(&mut self.texture_atlas); +    pub fn trim_cache( +        &mut self, +        device: &wgpu::Device, +        encoder: &mut wgpu::CommandEncoder, +    ) { +        #[cfg(feature = "image")] +        self.raster_cache +            .borrow_mut() +            .trim(&mut self.texture_atlas, &mut (device, encoder));          #[cfg(feature = "svg")] -        self.vector_cache.borrow_mut().trim(&mut self.texture_atlas); +        self.vector_cache +            .borrow_mut() +            .trim(&mut self.texture_atlas, &mut (device, encoder));      }  } @@ -509,15 +516,18 @@ fn add_instances(              add_instance(image_position, image_size, allocation, instances);          }          atlas::Entry::Fragmented { fragments, size } => { -            let scaling_x = image_size[0] / size.0 as f32; -            let scaling_y = image_size[1] / size.1 as f32; +            let scaling_x = image_size[0] / size.width as f32; +            let scaling_y = image_size[1] / size.height as f32;              for fragment in fragments {                  let allocation = &fragment.allocation;                  let [x, y] = image_position;                  let (fragment_x, fragment_y) = fragment.position; -                let (fragment_width, fragment_height) = allocation.size(); +                let Size { +                    width: fragment_width, +                    height: fragment_height, +                } = allocation.size();                  let position = [                      x + fragment_x as f32 * scaling_x, @@ -543,7 +553,7 @@ fn add_instance(      instances: &mut Vec<Instance>,  ) {      let (x, y) = allocation.position(); -    let (width, height) = allocation.size(); +    let Size { width, height } = allocation.size();      let layer = allocation.layer();      let instance = Instance { diff --git a/wgpu/src/image/atlas.rs b/wgpu/src/image/atlas.rs index 953dd4e2..eafe2f96 100644 --- a/wgpu/src/image/atlas.rs +++ b/wgpu/src/image/atlas.rs @@ -4,8 +4,6 @@ mod allocation;  mod allocator;  mod layer; -use std::num::NonZeroU32; -  pub use allocation::Allocation;  pub use entry::Entry;  pub use layer::Layer; @@ -14,6 +12,11 @@ use allocator::Allocator;  pub const SIZE: u32 = 2048; +use iced_graphics::image; +use iced_graphics::Size; + +use std::num::NonZeroU32; +  #[derive(Debug)]  pub struct Atlas {      texture: wgpu::Texture, @@ -35,7 +38,7 @@ impl Atlas {              mip_level_count: 1,              sample_count: 1,              dimension: wgpu::TextureDimension::D2, -            format: wgpu::TextureFormat::Bgra8UnormSrgb, +            format: wgpu::TextureFormat::Rgba8UnormSrgb,              usage: wgpu::TextureUsages::COPY_DST                  | wgpu::TextureUsages::COPY_SRC                  | wgpu::TextureUsages::TEXTURE_BINDING, @@ -61,99 +64,6 @@ impl Atlas {          self.layers.len()      } -    pub fn upload( -        &mut self, -        width: u32, -        height: u32, -        data: &[u8], -        device: &wgpu::Device, -        encoder: &mut wgpu::CommandEncoder, -    ) -> Option<Entry> { -        use wgpu::util::DeviceExt; - -        let entry = { -            let current_size = self.layers.len(); -            let entry = self.allocate(width, height)?; - -            // We grow the internal texture after allocating if necessary -            let new_layers = self.layers.len() - current_size; -            self.grow(new_layers, device, encoder); - -            entry -        }; - -        log::info!("Allocated atlas entry: {:?}", entry); - -        // It is a webgpu requirement that: -        //   BufferCopyView.layout.bytes_per_row % wgpu::COPY_BYTES_PER_ROW_ALIGNMENT == 0 -        // So we calculate padded_width by rounding width up to the next -        // multiple of wgpu::COPY_BYTES_PER_ROW_ALIGNMENT. -        let align = wgpu::COPY_BYTES_PER_ROW_ALIGNMENT; -        let padding = (align - (4 * width) % align) % align; -        let padded_width = (4 * width + padding) as usize; -        let padded_data_size = padded_width * height as usize; - -        let mut padded_data = vec![0; padded_data_size]; - -        for row in 0..height as usize { -            let offset = row * padded_width; - -            padded_data[offset..offset + 4 * width as usize].copy_from_slice( -                &data[row * 4 * width as usize..(row + 1) * 4 * width as usize], -            ) -        } - -        let buffer = -            device.create_buffer_init(&wgpu::util::BufferInitDescriptor { -                label: Some("iced_wgpu::image staging buffer"), -                contents: &padded_data, -                usage: wgpu::BufferUsages::COPY_SRC, -            }); - -        match &entry { -            Entry::Contiguous(allocation) => { -                self.upload_allocation( -                    &buffer, width, height, padding, 0, allocation, encoder, -                ); -            } -            Entry::Fragmented { fragments, .. } => { -                for fragment in fragments { -                    let (x, y) = fragment.position; -                    let offset = (y * padded_width as u32 + 4 * x) as usize; - -                    self.upload_allocation( -                        &buffer, -                        width, -                        height, -                        padding, -                        offset, -                        &fragment.allocation, -                        encoder, -                    ); -                } -            } -        } - -        log::info!("Current atlas: {:?}", self); - -        Some(entry) -    } - -    pub fn remove(&mut self, entry: &Entry) { -        log::info!("Removing atlas entry: {:?}", entry); - -        match entry { -            Entry::Contiguous(allocation) => { -                self.deallocate(allocation); -            } -            Entry::Fragmented { fragments, .. } => { -                for fragment in fragments { -                    self.deallocate(&fragment.allocation); -                } -            } -        } -    } -      fn allocate(&mut self, width: u32, height: u32) -> Option<Entry> {          // Allocate one layer if texture fits perfectly          if width == SIZE && height == SIZE { @@ -204,7 +114,7 @@ impl Atlas {              }              return Some(Entry::Fragmented { -                size: (width, height), +                size: Size::new(width, height),                  fragments,              });          } @@ -284,7 +194,7 @@ impl Atlas {          encoder: &mut wgpu::CommandEncoder,      ) {          let (x, y) = allocation.position(); -        let (width, height) = allocation.size(); +        let Size { width, height } = allocation.size();          let layer = allocation.layer();          let extent = wgpu::Extent3d { @@ -336,7 +246,7 @@ impl Atlas {              mip_level_count: 1,              sample_count: 1,              dimension: wgpu::TextureDimension::D2, -            format: wgpu::TextureFormat::Bgra8UnormSrgb, +            format: wgpu::TextureFormat::Rgba8UnormSrgb,              usage: wgpu::TextureUsages::COPY_DST                  | wgpu::TextureUsages::COPY_SRC                  | wgpu::TextureUsages::TEXTURE_BINDING, @@ -388,3 +298,100 @@ impl Atlas {              });      }  } + +impl image::Storage for Atlas { +    type Entry = Entry; +    type State<'a> = (&'a wgpu::Device, &'a mut wgpu::CommandEncoder); + +    fn upload( +        &mut self, +        width: u32, +        height: u32, +        data: &[u8], +        (device, encoder): &mut Self::State<'_>, +    ) -> Option<Self::Entry> { +        use wgpu::util::DeviceExt; + +        let entry = { +            let current_size = self.layers.len(); +            let entry = self.allocate(width, height)?; + +            // We grow the internal texture after allocating if necessary +            let new_layers = self.layers.len() - current_size; +            self.grow(new_layers, device, encoder); + +            entry +        }; + +        log::info!("Allocated atlas entry: {:?}", entry); + +        // It is a webgpu requirement that: +        //   BufferCopyView.layout.bytes_per_row % wgpu::COPY_BYTES_PER_ROW_ALIGNMENT == 0 +        // So we calculate padded_width by rounding width up to the next +        // multiple of wgpu::COPY_BYTES_PER_ROW_ALIGNMENT. +        let align = wgpu::COPY_BYTES_PER_ROW_ALIGNMENT; +        let padding = (align - (4 * width) % align) % align; +        let padded_width = (4 * width + padding) as usize; +        let padded_data_size = padded_width * height as usize; + +        let mut padded_data = vec![0; padded_data_size]; + +        for row in 0..height as usize { +            let offset = row * padded_width; + +            padded_data[offset..offset + 4 * width as usize].copy_from_slice( +                &data[row * 4 * width as usize..(row + 1) * 4 * width as usize], +            ) +        } + +        let buffer = +            device.create_buffer_init(&wgpu::util::BufferInitDescriptor { +                label: Some("iced_wgpu::image staging buffer"), +                contents: &padded_data, +                usage: wgpu::BufferUsages::COPY_SRC, +            }); + +        match &entry { +            Entry::Contiguous(allocation) => { +                self.upload_allocation( +                    &buffer, width, height, padding, 0, allocation, encoder, +                ); +            } +            Entry::Fragmented { fragments, .. } => { +                for fragment in fragments { +                    let (x, y) = fragment.position; +                    let offset = (y * padded_width as u32 + 4 * x) as usize; + +                    self.upload_allocation( +                        &buffer, +                        width, +                        height, +                        padding, +                        offset, +                        &fragment.allocation, +                        encoder, +                    ); +                } +            } +        } + +        log::info!("Current atlas: {:?}", self); + +        Some(entry) +    } + +    fn remove(&mut self, entry: &Entry, _: &mut Self::State<'_>) { +        log::info!("Removing atlas entry: {:?}", entry); + +        match entry { +            Entry::Contiguous(allocation) => { +                self.deallocate(allocation); +            } +            Entry::Fragmented { fragments, .. } => { +                for fragment in fragments { +                    self.deallocate(&fragment.allocation); +                } +            } +        } +    } +} diff --git a/wgpu/src/image/atlas/allocation.rs b/wgpu/src/image/atlas/allocation.rs index 59b7239f..43aba875 100644 --- a/wgpu/src/image/atlas/allocation.rs +++ b/wgpu/src/image/atlas/allocation.rs @@ -1,5 +1,7 @@  use crate::image::atlas::{self, allocator}; +use iced_graphics::Size; +  #[derive(Debug)]  pub enum Allocation {      Partial { @@ -19,10 +21,10 @@ impl Allocation {          }      } -    pub fn size(&self) -> (u32, u32) { +    pub fn size(&self) -> Size<u32> {          match self {              Allocation::Partial { region, .. } => region.size(), -            Allocation::Full { .. } => (atlas::SIZE, atlas::SIZE), +            Allocation::Full { .. } => Size::new(atlas::SIZE, atlas::SIZE),          }      } diff --git a/wgpu/src/image/atlas/allocator.rs b/wgpu/src/image/atlas/allocator.rs index 7a4ff5b1..03effdcb 100644 --- a/wgpu/src/image/atlas/allocator.rs +++ b/wgpu/src/image/atlas/allocator.rs @@ -46,10 +46,10 @@ impl Region {          (rectangle.min.x as u32, rectangle.min.y as u32)      } -    pub fn size(&self) -> (u32, u32) { +    pub fn size(&self) -> iced_graphics::Size<u32> {          let size = self.allocation.rectangle.size(); -        (size.width as u32, size.height as u32) +        iced_graphics::Size::new(size.width as u32, size.height as u32)      }  } diff --git a/wgpu/src/image/atlas/entry.rs b/wgpu/src/image/atlas/entry.rs index 9b3f16df..69c05a50 100644 --- a/wgpu/src/image/atlas/entry.rs +++ b/wgpu/src/image/atlas/entry.rs @@ -1,17 +1,19 @@  use crate::image::atlas; +use iced_graphics::image; +use iced_graphics::Size; +  #[derive(Debug)]  pub enum Entry {      Contiguous(atlas::Allocation),      Fragmented { -        size: (u32, u32), +        size: Size<u32>,          fragments: Vec<Fragment>,      },  } -impl Entry { -    #[cfg(feature = "image_rs")] -    pub fn size(&self) -> (u32, u32) { +impl image::storage::Entry for Entry { +    fn size(&self) -> Size<u32> {          match self {              Entry::Contiguous(allocation) => allocation.size(),              Entry::Fragmented { size, .. } => *size, | 
