diff options
| author | 2024-05-08 19:16:20 +0900 | |
|---|---|---|
| committer | 2024-05-08 19:16:20 +0900 | |
| commit | ff26fb7df43bd241253af85e0621e2cc030e9cec (patch) | |
| tree | 2b3272bb178f8757169511966589fe6a106733f4 /graphics | |
| parent | 0ebe0629cef37aee5c48b9409fc36618a3a3e60d (diff) | |
| parent | 477887b3870aa5fbdab96c3a06f3b930462d7842 (diff) | |
| download | iced-ff26fb7df43bd241253af85e0621e2cc030e9cec.tar.gz iced-ff26fb7df43bd241253af85e0621e2cc030e9cec.tar.bz2 iced-ff26fb7df43bd241253af85e0621e2cc030e9cec.zip | |
Merge branch 'iced-rs-master' into viewer_content_fit
Diffstat (limited to 'graphics')
| -rw-r--r-- | graphics/src/cache.rs | 189 | ||||
| -rw-r--r-- | graphics/src/cached.rs | 24 | ||||
| -rw-r--r-- | graphics/src/geometry.rs | 2 | ||||
| -rw-r--r-- | graphics/src/geometry/cache.rs | 83 | ||||
| -rw-r--r-- | graphics/src/image.rs | 75 | ||||
| -rw-r--r-- | graphics/src/lib.rs | 4 | ||||
| -rw-r--r-- | graphics/src/text/editor.rs | 12 | 
7 files changed, 289 insertions, 100 deletions
| diff --git a/graphics/src/cache.rs b/graphics/src/cache.rs new file mode 100644 index 00000000..bbba79eb --- /dev/null +++ b/graphics/src/cache.rs @@ -0,0 +1,189 @@ +//! Cache computations and efficiently reuse them. +use std::cell::RefCell; +use std::fmt; +use std::sync::atomic::{self, AtomicU64}; + +/// A simple cache that stores generated values to avoid recomputation. +/// +/// Keeps track of the last generated value after clearing. +pub struct Cache<T> { +    group: Group, +    state: RefCell<State<T>>, +} + +impl<T> Cache<T> { +    /// Creates a new empty [`Cache`]. +    pub fn new() -> Self { +        Cache { +            group: Group::singleton(), +            state: RefCell::new(State::Empty { previous: None }), +        } +    } + +    /// Creates a new empty [`Cache`] with the given [`Group`]. +    /// +    /// Caches within the same group may reuse internal rendering storage. +    /// +    /// You should generally group caches that are likely to change +    /// together. +    pub fn with_group(group: Group) -> Self { +        assert!( +            !group.is_singleton(), +            "The group {group:?} cannot be shared!" +        ); + +        Cache { +            group, +            state: RefCell::new(State::Empty { previous: None }), +        } +    } + +    /// Returns the [`Group`] of the [`Cache`]. +    pub fn group(&self) -> Group { +        self.group +    } + +    /// Puts the given value in the [`Cache`]. +    /// +    /// Notice that, given this is a cache, a mutable reference is not +    /// necessary to call this method. You can safely update the cache in +    /// rendering code. +    pub fn put(&self, value: T) { +        *self.state.borrow_mut() = State::Filled { current: value }; +    } + +    /// Returns a reference cell to the internal [`State`] of the [`Cache`]. +    pub fn state(&self) -> &RefCell<State<T>> { +        &self.state +    } + +    /// Clears the [`Cache`]. +    pub fn clear(&self) +    where +        T: Clone, +    { +        use std::ops::Deref; + +        let previous = match self.state.borrow().deref() { +            State::Empty { previous } => previous.clone(), +            State::Filled { current } => Some(current.clone()), +        }; + +        *self.state.borrow_mut() = State::Empty { previous }; +    } +} + +/// A cache group. +/// +/// Caches that share the same group generally change together. +/// +/// A cache group can be used to implement certain performance +/// optimizations during rendering, like batching or sharing atlases. +#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)] +pub struct Group { +    id: u64, +    is_singleton: bool, +} + +impl Group { +    /// Generates a new unique cache [`Group`]. +    pub fn unique() -> Self { +        static NEXT: AtomicU64 = AtomicU64::new(0); + +        Self { +            id: NEXT.fetch_add(1, atomic::Ordering::Relaxed), +            is_singleton: false, +        } +    } + +    /// Returns `true` if the [`Group`] can only ever have a +    /// single [`Cache`] in it. +    /// +    /// This is the default kind of [`Group`] assigned when using +    /// [`Cache::new`]. +    /// +    /// Knowing that a [`Group`] will never be shared may be +    /// useful for rendering backends to perform additional +    /// optimizations. +    pub fn is_singleton(self) -> bool { +        self.is_singleton +    } + +    fn singleton() -> Self { +        Self { +            is_singleton: true, +            ..Self::unique() +        } +    } +} + +impl<T> fmt::Debug for Cache<T> +where +    T: fmt::Debug, +{ +    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { +        use std::ops::Deref; + +        let state = self.state.borrow(); + +        match state.deref() { +            State::Empty { previous } => { +                write!(f, "Cache::Empty {{ previous: {previous:?} }}") +            } +            State::Filled { current } => { +                write!(f, "Cache::Filled {{ current: {current:?} }}") +            } +        } +    } +} + +impl<T> Default for Cache<T> { +    fn default() -> Self { +        Self::new() +    } +} + +/// The state of a [`Cache`]. +#[derive(Debug, Clone, Copy, PartialEq, Eq)] +pub enum State<T> { +    /// The [`Cache`] is empty. +    Empty { +        /// The previous value of the [`Cache`]. +        previous: Option<T>, +    }, +    /// The [`Cache`] is filled. +    Filled { +        /// The current value of the [`Cache`] +        current: T, +    }, +} + +/// A piece of data that can be cached. +pub trait Cached: Sized { +    /// The type of cache produced. +    type Cache: Clone; + +    /// Loads the [`Cache`] into a proper instance. +    /// +    /// [`Cache`]: Self::Cache +    fn load(cache: &Self::Cache) -> Self; + +    /// Caches this value, producing its corresponding [`Cache`]. +    /// +    /// [`Cache`]: Self::Cache +    fn cache(self, group: Group, previous: Option<Self::Cache>) -> Self::Cache; +} + +#[cfg(debug_assertions)] +impl Cached for () { +    type Cache = (); + +    fn load(_cache: &Self::Cache) -> Self {} + +    fn cache( +        self, +        _group: Group, +        _previous: Option<Self::Cache>, +    ) -> Self::Cache { +    } +} diff --git a/graphics/src/cached.rs b/graphics/src/cached.rs deleted file mode 100644 index c0e4e029..00000000 --- a/graphics/src/cached.rs +++ /dev/null @@ -1,24 +0,0 @@ -/// A piece of data that can be cached. -pub trait Cached: Sized { -    /// The type of cache produced. -    type Cache: Clone; - -    /// Loads the [`Cache`] into a proper instance. -    /// -    /// [`Cache`]: Self::Cache -    fn load(cache: &Self::Cache) -> Self; - -    /// Caches this value, producing its corresponding [`Cache`]. -    /// -    /// [`Cache`]: Self::Cache -    fn cache(self, previous: Option<Self::Cache>) -> Self::Cache; -} - -#[cfg(debug_assertions)] -impl Cached for () { -    type Cache = (); - -    fn load(_cache: &Self::Cache) -> Self {} - -    fn cache(self, _previous: Option<Self::Cache>) -> Self::Cache {} -} diff --git a/graphics/src/geometry.rs b/graphics/src/geometry.rs index dd07097f..ab4a7a36 100644 --- a/graphics/src/geometry.rs +++ b/graphics/src/geometry.rs @@ -18,8 +18,8 @@ pub use text::Text;  pub use crate::gradient::{self, Gradient}; +use crate::cache::Cached;  use crate::core::{self, Size}; -use crate::Cached;  /// A renderer capable of drawing some [`Self::Geometry`].  pub trait Renderer: core::Renderer { diff --git a/graphics/src/geometry/cache.rs b/graphics/src/geometry/cache.rs index 665e996b..d70cee0b 100644 --- a/graphics/src/geometry/cache.rs +++ b/graphics/src/geometry/cache.rs @@ -1,8 +1,8 @@ +use crate::cache::{self, Cached};  use crate::core::Size;  use crate::geometry::{self, Frame}; -use crate::Cached; -use std::cell::RefCell; +pub use cache::Group;  /// A simple cache that stores generated geometry to avoid recomputation.  /// @@ -12,7 +12,13 @@ pub struct Cache<Renderer>  where      Renderer: geometry::Renderer,  { -    state: RefCell<State<Renderer::Geometry>>, +    raw: crate::Cache<Data<<Renderer::Geometry as Cached>::Cache>>, +} + +#[derive(Debug, Clone)] +struct Data<T> { +    bounds: Size, +    geometry: T,  }  impl<Renderer> Cache<Renderer> @@ -22,20 +28,25 @@ where      /// Creates a new empty [`Cache`].      pub fn new() -> Self {          Cache { -            state: RefCell::new(State::Empty { previous: None }), +            raw: cache::Cache::new(), +        } +    } + +    /// Creates a new empty [`Cache`] with the given [`Group`]. +    /// +    /// Caches within the same group may reuse internal rendering storage. +    /// +    /// You should generally group caches that are likely to change +    /// together. +    pub fn with_group(group: Group) -> Self { +        Cache { +            raw: crate::Cache::with_group(group),          }      }      /// Clears the [`Cache`], forcing a redraw the next time it is used.      pub fn clear(&self) { -        use std::ops::Deref; - -        let previous = match self.state.borrow().deref() { -            State::Empty { previous } => previous.clone(), -            State::Filled { geometry, .. } => Some(geometry.clone()), -        }; - -        *self.state.borrow_mut() = State::Empty { previous }; +        self.raw.clear();      }      /// Draws geometry using the provided closure and stores it in the @@ -56,27 +67,30 @@ where      ) -> Renderer::Geometry {          use std::ops::Deref; -        let previous = match self.state.borrow().deref() { -            State::Empty { previous } => previous.clone(), -            State::Filled { -                bounds: cached_bounds, -                geometry, -            } => { -                if *cached_bounds == bounds { -                    return Cached::load(geometry); +        let state = self.raw.state(); + +        let previous = match state.borrow().deref() { +            cache::State::Empty { previous } => { +                previous.as_ref().map(|data| data.geometry.clone()) +            } +            cache::State::Filled { current } => { +                if current.bounds == bounds { +                    return Cached::load(¤t.geometry);                  } -                Some(geometry.clone()) +                Some(current.geometry.clone())              }          };          let mut frame = Frame::new(renderer, bounds);          draw_fn(&mut frame); -        let geometry = frame.into_geometry().cache(previous); +        let geometry = frame.into_geometry().cache(self.raw.group(), previous);          let result = Cached::load(&geometry); -        *self.state.borrow_mut() = State::Filled { bounds, geometry }; +        *state.borrow_mut() = cache::State::Filled { +            current: Data { bounds, geometry }, +        };          result      } @@ -85,16 +99,10 @@ where  impl<Renderer> std::fmt::Debug for Cache<Renderer>  where      Renderer: geometry::Renderer, +    <Renderer::Geometry as Cached>::Cache: std::fmt::Debug,  {      fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { -        let state = self.state.borrow(); - -        match *state { -            State::Empty { .. } => write!(f, "Cache::Empty"), -            State::Filled { bounds, .. } => { -                write!(f, "Cache::Filled {{ bounds: {bounds:?} }}") -            } -        } +        write!(f, "{:?}", &self.raw)      }  } @@ -106,16 +114,3 @@ where          Self::new()      }  } - -enum State<Geometry> -where -    Geometry: Cached, -{ -    Empty { -        previous: Option<Geometry::Cache>, -    }, -    Filled { -        bounds: Size, -        geometry: Geometry::Cache, -    }, -} diff --git a/graphics/src/image.rs b/graphics/src/image.rs index c6135e9e..318592be 100644 --- a/graphics/src/image.rs +++ b/graphics/src/image.rs @@ -2,9 +2,7 @@  #[cfg(feature = "image")]  pub use ::image as image_rs; -use crate::core::image; -use crate::core::svg; -use crate::core::{Color, Rectangle}; +use crate::core::{image, svg, Color, Radians, Rectangle};  /// A raster or vector image.  #[derive(Debug, Clone, PartialEq)] @@ -19,6 +17,12 @@ pub enum Image {          /// The bounds of the image.          bounds: Rectangle, + +        /// The rotation of the image. +        rotation: Radians, + +        /// The opacity of the image. +        opacity: f32,      },      /// A vector image.      Vector { @@ -30,6 +34,12 @@ pub enum Image {          /// The bounds of the image.          bounds: Rectangle, + +        /// The rotation of the image. +        rotation: Radians, + +        /// The opacity of the image. +        opacity: f32,      },  } @@ -37,9 +47,12 @@ impl Image {      /// Returns the bounds of the [`Image`].      pub fn bounds(&self) -> Rectangle {          match self { -            Image::Raster { bounds, .. } | Image::Vector { bounds, .. } => { -                *bounds +            Image::Raster { +                bounds, rotation, ..              } +            | Image::Vector { +                bounds, rotation, .. +            } => bounds.rotate(*rotation),          }      }  } @@ -50,7 +63,8 @@ impl Image {  /// [`Handle`]: image::Handle  pub fn load(      handle: &image::Handle, -) -> ::image::ImageResult<::image::DynamicImage> { +) -> ::image::ImageResult<::image::ImageBuffer<::image::Rgba<u8>, image::Bytes>> +{      use bitflags::bitflags;      bitflags! { @@ -100,8 +114,8 @@ pub fn load(          }      } -    match handle.data() { -        image::Data::Path(path) => { +    let (width, height, pixels) = match handle { +        image::Handle::Path(_, path) => {              let image = ::image::open(path)?;              let operation = std::fs::File::open(path) @@ -110,33 +124,44 @@ pub fn load(                  .and_then(|mut reader| Operation::from_exif(&mut reader).ok())                  .unwrap_or_else(Operation::empty); -            Ok(operation.perform(image)) +            let rgba = operation.perform(image).into_rgba8(); + +            ( +                rgba.width(), +                rgba.height(), +                image::Bytes::from(rgba.into_raw()), +            )          } -        image::Data::Bytes(bytes) => { +        image::Handle::Bytes(_, bytes) => {              let image = ::image::load_from_memory(bytes)?;              let operation =                  Operation::from_exif(&mut std::io::Cursor::new(bytes))                      .ok()                      .unwrap_or_else(Operation::empty); -            Ok(operation.perform(image)) +            let rgba = operation.perform(image).into_rgba8(); + +            ( +                rgba.width(), +                rgba.height(), +                image::Bytes::from(rgba.into_raw()), +            )          } -        image::Data::Rgba { +        image::Handle::Rgba {              width,              height,              pixels, -        } => { -            if let Some(image) = -                ::image::ImageBuffer::from_vec(*width, *height, pixels.to_vec()) -            { -                Ok(::image::DynamicImage::ImageRgba8(image)) -            } else { -                Err(::image::error::ImageError::Limits( -                    ::image::error::LimitError::from_kind( -                        ::image::error::LimitErrorKind::DimensionError, -                    ), -                )) -            } -        } +            .. +        } => (*width, *height, pixels.clone()), +    }; + +    if let Some(image) = ::image::ImageBuffer::from_raw(width, height, pixels) { +        Ok(image) +    } else { +        Err(::image::error::ImageError::Limits( +            ::image::error::LimitError::from_kind( +                ::image::error::LimitErrorKind::DimensionError, +            ), +        ))      }  } diff --git a/graphics/src/lib.rs b/graphics/src/lib.rs index 865ebd97..b5ef55e7 100644 --- a/graphics/src/lib.rs +++ b/graphics/src/lib.rs @@ -9,10 +9,10 @@  )]  #![cfg_attr(docsrs, feature(doc_auto_cfg))]  mod antialiasing; -mod cached;  mod settings;  mod viewport; +pub mod cache;  pub mod color;  pub mod compositor;  pub mod damage; @@ -27,7 +27,7 @@ pub mod text;  pub mod geometry;  pub use antialiasing::Antialiasing; -pub use cached::Cached; +pub use cache::Cache;  pub use compositor::Compositor;  pub use error::Error;  pub use gradient::Gradient; diff --git a/graphics/src/text/editor.rs b/graphics/src/text/editor.rs index c488a51c..4b8f0f2a 100644 --- a/graphics/src/text/editor.rs +++ b/graphics/src/text/editor.rs @@ -456,10 +456,14 @@ impl editor::Editor for Editor {                  }              }              Action::Scroll { lines } => { -                editor.action( -                    font_system.raw(), -                    cosmic_text::Action::Scroll { lines }, -                ); +                let (_, height) = editor.buffer().size(); + +                if height < i32::MAX as f32 { +                    editor.action( +                        font_system.raw(), +                        cosmic_text::Action::Scroll { lines }, +                    ); +                }              }          } | 
