diff options
| author | 2019-11-29 21:44:39 +0100 | |
|---|---|---|
| committer | 2019-11-29 21:44:39 +0100 | |
| commit | 505588d5851fed8a3bb334edacfa6d96c545d562 (patch) | |
| tree | 1fb107d7c5f9bae0b6fe0c9ff106d7b9bd483b52 /native/src/widget | |
| parent | 811d8b90d71c26100f0933217f5474e090fbf17c (diff) | |
| download | iced-505588d5851fed8a3bb334edacfa6d96c545d562.tar.gz iced-505588d5851fed8a3bb334edacfa6d96c545d562.tar.bz2 iced-505588d5851fed8a3bb334edacfa6d96c545d562.zip | |
Allow to load an image from memory
New `image::Handle` opaque type uniquely identifying some `image::Data`,
allowing reliable caching.
Diffstat (limited to 'native/src/widget')
| -rw-r--r-- | native/src/widget/image.rs | 93 | 
1 files changed, 85 insertions, 8 deletions
| diff --git a/native/src/widget/image.rs b/native/src/widget/image.rs index 4c588c9d..c82525e4 100644 --- a/native/src/widget/image.rs +++ b/native/src/widget/image.rs @@ -2,7 +2,11 @@  use crate::{layout, Element, Hasher, Layout, Length, Point, Size, Widget}; -use std::hash::Hash; +use std::{ +    hash::{Hash, Hasher as _}, +    path::PathBuf, +    rc::Rc, +};  /// A frame that displays an image while keeping aspect ratio.  /// @@ -17,7 +21,7 @@ use std::hash::Hash;  /// <img src="https://github.com/hecrj/iced/blob/9712b319bb7a32848001b96bd84977430f14b623/examples/resources/ferris.png?raw=true" width="300">  #[derive(Debug)]  pub struct Image { -    path: String, +    handle: Handle,      width: Length,      height: Length,  } @@ -26,9 +30,9 @@ impl Image {      /// Creates a new [`Image`] with the given path.      ///      /// [`Image`]: struct.Image.html -    pub fn new<T: Into<String>>(path: T) -> Self { +    pub fn new<T: Into<Handle>>(handle: T) -> Self {          Image { -            path: path.into(), +            handle: handle.into(),              width: Length::Shrink,              height: Length::Shrink,          } @@ -68,7 +72,7 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        let (width, height) = renderer.dimensions(&self.path); +        let (width, height) = renderer.dimensions(&self.handle);          let aspect_ratio = width as f32 / height as f32; @@ -96,7 +100,7 @@ where          layout: Layout<'_>,          _cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self.path, layout) +        renderer.draw(self.handle.clone(), layout)      }      fn hash_layout(&self, state: &mut Hasher) { @@ -105,6 +109,79 @@ where      }  } +/// An [`Image`] handle. +/// +/// [`Image`]: struct.Image.html +#[derive(Debug, Clone)] +pub struct Handle { +    id: u64, +    data: Rc<Data>, +} + +impl Handle { +    /// Creates an image [`Handle`] pointing to the image of the given path. +    /// +    /// [`Handle`]: struct.Handle.html +    pub fn from_path<T: Into<PathBuf>>(path: T) -> Handle { +        Self::from_data(Data::Path(path.into())) +    } + +    /// Creates an image [`Handle`] containing the image data directly. +    /// +    /// [`Handle`]: struct.Handle.html +    pub fn from_bytes(bytes: Vec<u8>) -> Handle { +        Self::from_data(Data::Bytes(bytes)) +    } + +    fn from_data(data: Data) -> Handle { +        let mut hasher = Hasher::default(); +        data.hash(&mut hasher); + +        Handle { +            id: hasher.finish(), +            data: Rc::new(data), +        } +    } + +    /// Returns the uniquie identifier of the [`Handle`]. +    /// +    /// [`Handle`]: struct.Handle.html +    pub fn id(&self) -> u64 { +        self.id +    } + +    /// Returns a reference to the image [`Data`]. +    /// +    /// [`Data`]: enum.Data.html +    pub fn data(&self) -> &Data { +        &self.data +    } +} + +impl From<String> for Handle { +    fn from(path: String) -> Handle { +        Handle::from_path(path) +    } +} + +impl From<&str> for Handle { +    fn from(path: &str) -> Handle { +        Handle::from_path(path) +    } +} + +/// The data of an [`Image`]. +/// +/// [`Image`]: struct.Image.html +#[derive(Debug, Clone, Hash)] +pub enum Data { +    /// File data +    Path(PathBuf), + +    /// In-memory data +    Bytes(Vec<u8>), +} +  /// The renderer of an [`Image`].  ///  /// Your [renderer] will need to implement this trait before being able to use @@ -116,12 +193,12 @@ pub trait Renderer: crate::Renderer {      /// Returns the dimensions of an [`Image`] located on the given path.      ///      /// [`Image`]: struct.Image.html -    fn dimensions(&self, path: &str) -> (u32, u32); +    fn dimensions(&self, handle: &Handle) -> (u32, u32);      /// Draws an [`Image`].      ///      /// [`Image`]: struct.Image.html -    fn draw(&mut self, path: &str, layout: Layout<'_>) -> Self::Output; +    fn draw(&mut self, handle: Handle, layout: Layout<'_>) -> Self::Output;  }  impl<'a, Message, Renderer> From<Image> for Element<'a, Message, Renderer> | 
