diff options
Diffstat (limited to 'graphics/src')
| -rw-r--r-- | graphics/src/backend.rs | 36 | ||||
| -rw-r--r-- | graphics/src/cached.rs | 24 | ||||
| -rw-r--r-- | graphics/src/compositor.rs | 14 | ||||
| -rw-r--r-- | graphics/src/damage.rs | 240 | ||||
| -rw-r--r-- | graphics/src/geometry.rs | 9 | ||||
| -rw-r--r-- | graphics/src/geometry/cache.rs | 39 | ||||
| -rw-r--r-- | graphics/src/geometry/frame.rs | 16 | ||||
| -rw-r--r-- | graphics/src/image.rs | 167 | ||||
| -rw-r--r-- | graphics/src/layer.rs | 144 | ||||
| -rw-r--r-- | graphics/src/lib.rs | 23 | ||||
| -rw-r--r-- | graphics/src/mesh.rs | 94 | ||||
| -rw-r--r-- | graphics/src/primitive.rs | 160 | ||||
| -rw-r--r-- | graphics/src/renderer.rs | 269 | ||||
| -rw-r--r-- | graphics/src/settings.rs | 2 | ||||
| -rw-r--r-- | graphics/src/text.rs | 129 | ||||
| -rw-r--r-- | graphics/src/text/cache.rs | 16 | ||||
| -rw-r--r-- | graphics/src/text/paragraph.rs | 4 | 
17 files changed, 554 insertions, 832 deletions
| diff --git a/graphics/src/backend.rs b/graphics/src/backend.rs deleted file mode 100644 index 7abc42c5..00000000 --- a/graphics/src/backend.rs +++ /dev/null @@ -1,36 +0,0 @@ -//! Write a graphics backend. -use crate::core::image; -use crate::core::svg; -use crate::core::Size; -use crate::{Compositor, Mesh, Renderer}; - -use std::borrow::Cow; - -/// The graphics backend of a [`Renderer`]. -/// -/// [`Renderer`]: crate::Renderer -pub trait Backend: Sized { -    /// The custom kind of primitives this [`Backend`] supports. -    type Primitive: TryFrom<Mesh, Error = &'static str>; - -    /// The default compositor of this [`Backend`]. -    type Compositor: Compositor<Renderer = Renderer<Self>>; -} - -/// A graphics backend that supports text rendering. -pub trait Text { -    /// Loads a font from its bytes. -    fn load_font(&mut self, font: Cow<'static, [u8]>); -} - -/// A graphics backend that supports image rendering. -pub trait Image { -    /// Returns the dimensions of the provided image. -    fn dimensions(&self, handle: &image::Handle) -> Size<u32>; -} - -/// A graphics backend that supports SVG rendering. -pub trait Svg { -    /// Returns the viewport dimensions of the provided SVG. -    fn viewport_dimensions(&self, handle: &svg::Handle) -> Size<u32>; -} diff --git a/graphics/src/cached.rs b/graphics/src/cached.rs index b52f9d9d..c0e4e029 100644 --- a/graphics/src/cached.rs +++ b/graphics/src/cached.rs @@ -1,11 +1,7 @@ -use crate::Primitive; - -use std::sync::Arc; -  /// A piece of data that can be cached.  pub trait Cached: Sized {      /// The type of cache produced. -    type Cache; +    type Cache: Clone;      /// Loads the [`Cache`] into a proper instance.      /// @@ -15,21 +11,7 @@ pub trait Cached: Sized {      /// Caches this value, producing its corresponding [`Cache`].      ///      /// [`Cache`]: Self::Cache -    fn cache(self) -> Self::Cache; -} - -impl<T> Cached for Primitive<T> { -    type Cache = Arc<Self>; - -    fn load(cache: &Arc<Self>) -> Self { -        Self::Cache { -            content: cache.clone(), -        } -    } - -    fn cache(self) -> Arc<Self> { -        Arc::new(self) -    } +    fn cache(self, previous: Option<Self::Cache>) -> Self::Cache;  }  #[cfg(debug_assertions)] @@ -38,5 +20,5 @@ impl Cached for () {      fn load(_cache: &Self::Cache) -> Self {} -    fn cache(self) -> Self::Cache {} +    fn cache(self, _previous: Option<Self::Cache>) -> Self::Cache {}  } diff --git a/graphics/src/compositor.rs b/graphics/src/compositor.rs index 86472a58..47521eb0 100644 --- a/graphics/src/compositor.rs +++ b/graphics/src/compositor.rs @@ -5,9 +5,11 @@ use crate::futures::{MaybeSend, MaybeSync};  use crate::{Error, Settings, Viewport};  use raw_window_handle::{HasDisplayHandle, HasWindowHandle}; -use std::future::Future;  use thiserror::Error; +use std::borrow::Cow; +use std::future::Future; +  /// A graphics compositor that can draw to windows.  pub trait Compositor: Sized {      /// The iced renderer of the backend. @@ -60,6 +62,14 @@ pub trait Compositor: Sized {      /// Returns [`Information`] used by this [`Compositor`].      fn fetch_information(&self) -> Information; +    /// Loads a font from its bytes. +    fn load_font(&mut self, font: Cow<'static, [u8]>) { +        crate::text::font_system() +            .write() +            .expect("Write to font system") +            .load_font(font); +    } +      /// Presents the [`Renderer`] primitives to the next frame of the given [`Surface`].      ///      /// [`Renderer`]: Self::Renderer @@ -168,6 +178,8 @@ impl Compositor for () {      ) {      } +    fn load_font(&mut self, _font: Cow<'static, [u8]>) {} +      fn fetch_information(&self) -> Information {          Information {              adapter: String::from("Null Renderer"), diff --git a/graphics/src/damage.rs b/graphics/src/damage.rs index 8edf69d7..17d60451 100644 --- a/graphics/src/damage.rs +++ b/graphics/src/damage.rs @@ -1,196 +1,14 @@ -//! Track and compute the damage of graphical primitives. -use crate::core::alignment; -use crate::core::{Rectangle, Size}; -use crate::Primitive; - -use std::sync::Arc; - -/// A type that has some damage bounds. -pub trait Damage: PartialEq { -    /// Returns the bounds of the [`Damage`]. -    fn bounds(&self) -> Rectangle; -} - -impl<T: Damage> Damage for Primitive<T> { -    fn bounds(&self) -> Rectangle { -        match self { -            Self::Text { -                bounds, -                horizontal_alignment, -                vertical_alignment, -                .. -            } => { -                let mut bounds = *bounds; - -                bounds.x = match horizontal_alignment { -                    alignment::Horizontal::Left => bounds.x, -                    alignment::Horizontal::Center => { -                        bounds.x - bounds.width / 2.0 -                    } -                    alignment::Horizontal::Right => bounds.x - bounds.width, -                }; - -                bounds.y = match vertical_alignment { -                    alignment::Vertical::Top => bounds.y, -                    alignment::Vertical::Center => { -                        bounds.y - bounds.height / 2.0 -                    } -                    alignment::Vertical::Bottom => bounds.y - bounds.height, -                }; - -                bounds.expand(1.5) -            } -            Self::Paragraph { -                paragraph, -                position, -                .. -            } => { -                let mut bounds = -                    Rectangle::new(*position, paragraph.min_bounds); - -                bounds.x = match paragraph.horizontal_alignment { -                    alignment::Horizontal::Left => bounds.x, -                    alignment::Horizontal::Center => { -                        bounds.x - bounds.width / 2.0 -                    } -                    alignment::Horizontal::Right => bounds.x - bounds.width, -                }; - -                bounds.y = match paragraph.vertical_alignment { -                    alignment::Vertical::Top => bounds.y, -                    alignment::Vertical::Center => { -                        bounds.y - bounds.height / 2.0 -                    } -                    alignment::Vertical::Bottom => bounds.y - bounds.height, -                }; - -                bounds.expand(1.5) -            } -            Self::Editor { -                editor, position, .. -            } => { -                let bounds = Rectangle::new(*position, editor.bounds); - -                bounds.expand(1.5) -            } -            Self::RawText(raw) => { -                // TODO: Add `size` field to `raw` to compute more accurate -                // damage bounds (?) -                raw.clip_bounds.expand(1.5) -            } -            Self::Quad { bounds, shadow, .. } if shadow.color.a > 0.0 => { -                let bounds_with_shadow = Rectangle { -                    x: bounds.x + shadow.offset.x.min(0.0) - shadow.blur_radius, -                    y: bounds.y + shadow.offset.y.min(0.0) - shadow.blur_radius, -                    width: bounds.width -                        + shadow.offset.x.abs() -                        + shadow.blur_radius * 2.0, -                    height: bounds.height -                        + shadow.offset.y.abs() -                        + shadow.blur_radius * 2.0, -                }; - -                bounds_with_shadow.expand(1.0) -            } -            Self::Quad { bounds, .. } -            | Self::Image { bounds, .. } -            | Self::Svg { bounds, .. } => bounds.expand(1.0), -            Self::Clip { bounds, .. } => bounds.expand(1.0), -            Self::Group { primitives } => primitives -                .iter() -                .map(Self::bounds) -                .fold(Rectangle::with_size(Size::ZERO), |a, b| { -                    Rectangle::union(&a, &b) -                }), -            Self::Transform { -                transformation, -                content, -            } => content.bounds() * *transformation, -            Self::Cache { content } => content.bounds(), -            Self::Custom(custom) => custom.bounds(), -        } -    } -} - -fn regions<T: Damage>(a: &Primitive<T>, b: &Primitive<T>) -> Vec<Rectangle> { -    match (a, b) { -        ( -            Primitive::Group { -                primitives: primitives_a, -            }, -            Primitive::Group { -                primitives: primitives_b, -            }, -        ) => return list(primitives_a, primitives_b), -        ( -            Primitive::Clip { -                bounds: bounds_a, -                content: content_a, -                .. -            }, -            Primitive::Clip { -                bounds: bounds_b, -                content: content_b, -                .. -            }, -        ) => { -            if bounds_a == bounds_b { -                return regions(content_a, content_b) -                    .into_iter() -                    .filter_map(|r| r.intersection(&bounds_a.expand(1.0))) -                    .collect(); -            } else { -                return vec![bounds_a.expand(1.0), bounds_b.expand(1.0)]; -            } -        } -        ( -            Primitive::Transform { -                transformation: transformation_a, -                content: content_a, -            }, -            Primitive::Transform { -                transformation: transformation_b, -                content: content_b, -            }, -        ) => { -            if transformation_a == transformation_b { -                return regions(content_a, content_b) -                    .into_iter() -                    .map(|r| r * *transformation_a) -                    .collect(); -            } -        } -        ( -            Primitive::Cache { content: content_a }, -            Primitive::Cache { content: content_b }, -        ) => { -            if Arc::ptr_eq(content_a, content_b) { -                return vec![]; -            } -        } -        _ if a == b => return vec![], -        _ => {} -    } - -    let bounds_a = a.bounds(); -    let bounds_b = b.bounds(); - -    if bounds_a == bounds_b { -        vec![bounds_a] -    } else { -        vec![bounds_a, bounds_b] -    } -} - -/// Computes the damage regions between the two given lists of primitives. -pub fn list<T: Damage>( -    previous: &[Primitive<T>], -    current: &[Primitive<T>], +//! Compute the damage between frames. +use crate::core::{Point, Rectangle}; + +/// Diffs the damage regions given some previous and current primitives. +pub fn diff<T>( +    previous: &[T], +    current: &[T], +    bounds: impl Fn(&T) -> Vec<Rectangle>, +    diff: impl Fn(&T, &T) -> Vec<Rectangle>,  ) -> Vec<Rectangle> { -    let damage = previous -        .iter() -        .zip(current) -        .flat_map(|(a, b)| regions(a, b)); +    let damage = previous.iter().zip(current).flat_map(|(a, b)| diff(a, b));      if previous.len() == current.len() {          damage.collect() @@ -203,39 +21,45 @@ pub fn list<T: Damage>(          // Extend damage by the added/removed primitives          damage -            .chain(bigger[smaller.len()..].iter().map(Damage::bounds)) +            .chain(bigger[smaller.len()..].iter().flat_map(bounds))              .collect()      }  } +/// Computes the damage regions given some previous and current primitives. +pub fn list<T>( +    previous: &[T], +    current: &[T], +    bounds: impl Fn(&T) -> Vec<Rectangle>, +    are_equal: impl Fn(&T, &T) -> bool, +) -> Vec<Rectangle> { +    diff(previous, current, &bounds, |a, b| { +        if are_equal(a, b) { +            vec![] +        } else { +            bounds(a).into_iter().chain(bounds(b)).collect() +        } +    }) +} +  /// Groups the given damage regions that are close together inside the given  /// bounds. -pub fn group( -    mut damage: Vec<Rectangle>, -    scale_factor: f32, -    bounds: Size<u32>, -) -> Vec<Rectangle> { +pub fn group(mut damage: Vec<Rectangle>, bounds: Rectangle) -> Vec<Rectangle> {      use std::cmp::Ordering;      const AREA_THRESHOLD: f32 = 20_000.0; -    let bounds = Rectangle { -        x: 0.0, -        y: 0.0, -        width: bounds.width as f32, -        height: bounds.height as f32, -    }; -      damage.sort_by(|a, b| { -        a.x.partial_cmp(&b.x) +        a.center() +            .distance(Point::ORIGIN) +            .partial_cmp(&b.center().distance(Point::ORIGIN))              .unwrap_or(Ordering::Equal) -            .then_with(|| a.y.partial_cmp(&b.y).unwrap_or(Ordering::Equal))      });      let mut output = Vec::new();      let mut scaled = damage          .into_iter() -        .filter_map(|region| (region * scale_factor).intersection(&bounds)) +        .filter_map(|region| region.intersection(&bounds))          .filter(|region| region.width >= 1.0 && region.height >= 1.0);      if let Some(mut current) = scaled.next() { diff --git a/graphics/src/geometry.rs b/graphics/src/geometry.rs index d251efb8..dd07097f 100644 --- a/graphics/src/geometry.rs +++ b/graphics/src/geometry.rs @@ -36,15 +36,6 @@ pub trait Renderer: core::Renderer {      fn draw_geometry(&mut self, geometry: Self::Geometry);  } -/// The graphics backend of a geometry renderer. -pub trait Backend { -    /// The kind of [`Frame`] this backend supports. -    type Frame: frame::Backend; - -    /// Creates a new [`Self::Frame`]. -    fn new_frame(&self, size: Size) -> Self::Frame; -} -  #[cfg(debug_assertions)]  impl Renderer for () {      type Geometry = (); diff --git a/graphics/src/geometry/cache.rs b/graphics/src/geometry/cache.rs index 37d433c2..665e996b 100644 --- a/graphics/src/geometry/cache.rs +++ b/graphics/src/geometry/cache.rs @@ -22,13 +22,20 @@ where      /// Creates a new empty [`Cache`].      pub fn new() -> Self {          Cache { -            state: RefCell::new(State::Empty), +            state: RefCell::new(State::Empty { previous: None }),          }      }      /// Clears the [`Cache`], forcing a redraw the next time it is used.      pub fn clear(&self) { -        *self.state.borrow_mut() = State::Empty; +        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 };      }      /// Draws geometry using the provided closure and stores it in the @@ -49,20 +56,24 @@ where      ) -> Renderer::Geometry {          use std::ops::Deref; -        if let State::Filled { -            bounds: cached_bounds, -            geometry, -        } = self.state.borrow().deref() -        { -            if *cached_bounds == bounds { -                return Cached::load(geometry); +        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); +                } + +                Some(geometry.clone())              } -        } +        };          let mut frame = Frame::new(renderer, bounds);          draw_fn(&mut frame); -        let geometry = frame.into_geometry().cache(); +        let geometry = frame.into_geometry().cache(previous);          let result = Cached::load(&geometry);          *self.state.borrow_mut() = State::Filled { bounds, geometry }; @@ -79,7 +90,7 @@ where          let state = self.state.borrow();          match *state { -            State::Empty => write!(f, "Cache::Empty"), +            State::Empty { .. } => write!(f, "Cache::Empty"),              State::Filled { bounds, .. } => {                  write!(f, "Cache::Filled {{ bounds: {bounds:?} }}")              } @@ -100,7 +111,9 @@ enum State<Geometry>  where      Geometry: Cached,  { -    Empty, +    Empty { +        previous: Option<Geometry::Cache>, +    },      Filled {          bounds: Size,          geometry: Geometry::Cache, diff --git a/graphics/src/geometry/frame.rs b/graphics/src/geometry/frame.rs index ad35e8ec..377589d7 100644 --- a/graphics/src/geometry/frame.rs +++ b/graphics/src/geometry/frame.rs @@ -113,13 +113,11 @@ where          region: Rectangle,          f: impl FnOnce(&mut Self) -> R,      ) -> R { -        let mut frame = self.draft(region.size()); +        let mut frame = self.draft(region);          let result = f(&mut frame); -        let origin = Point::new(region.x, region.y); - -        self.paste(frame, origin); +        self.paste(frame, Point::new(region.x, region.y));          result      } @@ -129,14 +127,14 @@ where      /// Draw its contents back to this [`Frame`] with [`paste`].      ///      /// [`paste`]: Self::paste -    pub fn draft(&mut self, size: Size) -> Self { +    fn draft(&mut self, clip_bounds: Rectangle) -> Self {          Self { -            raw: self.raw.draft(size), +            raw: self.raw.draft(clip_bounds),          }      }      /// Draws the contents of the given [`Frame`] with origin at the given [`Point`]. -    pub fn paste(&mut self, frame: Self, at: Point) { +    fn paste(&mut self, frame: Self, at: Point) {          self.raw.paste(frame.raw, at);      } @@ -187,7 +185,7 @@ pub trait Backend: Sized {      fn scale(&mut self, scale: impl Into<f32>);      fn scale_nonuniform(&mut self, scale: impl Into<Vector>); -    fn draft(&mut self, size: Size) -> Self; +    fn draft(&mut self, clip_bounds: Rectangle) -> Self;      fn paste(&mut self, frame: Self, at: Point);      fn stroke<'a>(&mut self, path: &Path, stroke: impl Into<Stroke<'a>>); @@ -232,7 +230,7 @@ impl Backend for () {      fn scale(&mut self, _scale: impl Into<f32>) {}      fn scale_nonuniform(&mut self, _scale: impl Into<Vector>) {} -    fn draft(&mut self, _size: Size) -> Self {} +    fn draft(&mut self, _clip_bounds: Rectangle) -> Self {}      fn paste(&mut self, _frame: Self, _at: Point) {}      fn stroke<'a>(&mut self, _path: &Path, _stroke: impl Into<Stroke<'a>>) {} diff --git a/graphics/src/image.rs b/graphics/src/image.rs index d89caace..c6135e9e 100644 --- a/graphics/src/image.rs +++ b/graphics/src/image.rs @@ -1,14 +1,107 @@  //! Load and operate on images. -use crate::core::image::{Data, Handle}; +#[cfg(feature = "image")] +pub use ::image as image_rs; -use bitflags::bitflags; +use crate::core::image; +use crate::core::svg; +use crate::core::{Color, Rectangle}; -pub use ::image as image_rs; +/// A raster or vector image. +#[derive(Debug, Clone, PartialEq)] +pub enum Image { +    /// A raster image. +    Raster { +        /// The handle of a raster image. +        handle: image::Handle, + +        /// The filter method of a raster image. +        filter_method: image::FilterMethod, + +        /// The bounds of the image. +        bounds: Rectangle, +    }, +    /// A vector image. +    Vector { +        /// The handle of a vector image. +        handle: svg::Handle, + +        /// The [`Color`] filter +        color: Option<Color>, + +        /// The bounds of the image. +        bounds: Rectangle, +    }, +} + +impl Image { +    /// Returns the bounds of the [`Image`]. +    pub fn bounds(&self) -> Rectangle { +        match self { +            Image::Raster { bounds, .. } | Image::Vector { bounds, .. } => { +                *bounds +            } +        } +    } +} +#[cfg(feature = "image")]  /// Tries to load an image by its [`Handle`]. -pub fn load(handle: &Handle) -> image_rs::ImageResult<image_rs::DynamicImage> { +/// +/// [`Handle`]: image::Handle +pub fn load( +    handle: &image::Handle, +) -> ::image::ImageResult<::image::DynamicImage> { +    use bitflags::bitflags; + +    bitflags! { +        struct Operation: u8 { +            const FLIP_HORIZONTALLY = 0b001; +            const ROTATE_180 = 0b010; +            const FLIP_DIAGONALLY = 0b100; +        } +    } + +    impl Operation { +        // Meaning of the returned value is described e.g. at: +        // https://magnushoff.com/articles/jpeg-orientation/ +        fn from_exif<R>(reader: &mut R) -> Result<Self, exif::Error> +        where +            R: std::io::BufRead + std::io::Seek, +        { +            let exif = exif::Reader::new().read_from_container(reader)?; + +            Ok(exif +                .get_field(exif::Tag::Orientation, exif::In::PRIMARY) +                .and_then(|field| field.value.get_uint(0)) +                .and_then(|value| u8::try_from(value).ok()) +                .and_then(|value| Self::from_bits(value.saturating_sub(1))) +                .unwrap_or_else(Self::empty)) +        } + +        fn perform( +            self, +            mut image: ::image::DynamicImage, +        ) -> ::image::DynamicImage { +            use ::image::imageops; + +            if self.contains(Self::FLIP_DIAGONALLY) { +                imageops::flip_vertical_in_place(&mut image); +            } + +            if self.contains(Self::ROTATE_180) { +                imageops::rotate180_in_place(&mut image); +            } + +            if self.contains(Self::FLIP_HORIZONTALLY) { +                imageops::flip_horizontal_in_place(&mut image); +            } + +            image +        } +    } +      match handle.data() { -        Data::Path(path) => { +        image::Data::Path(path) => {              let image = ::image::open(path)?;              let operation = std::fs::File::open(path) @@ -19,7 +112,7 @@ pub fn load(handle: &Handle) -> image_rs::ImageResult<image_rs::DynamicImage> {              Ok(operation.perform(image))          } -        Data::Bytes(bytes) => { +        image::Data::Bytes(bytes) => {              let image = ::image::load_from_memory(bytes)?;              let operation =                  Operation::from_exif(&mut std::io::Cursor::new(bytes)) @@ -28,68 +121,22 @@ pub fn load(handle: &Handle) -> image_rs::ImageResult<image_rs::DynamicImage> {              Ok(operation.perform(image))          } -        Data::Rgba { +        image::Data::Rgba {              width,              height,              pixels,          } => { -            if let Some(image) = image_rs::ImageBuffer::from_vec( -                *width, -                *height, -                pixels.to_vec(), -            ) { -                Ok(image_rs::DynamicImage::ImageRgba8(image)) +            if let Some(image) = +                ::image::ImageBuffer::from_vec(*width, *height, pixels.to_vec()) +            { +                Ok(::image::DynamicImage::ImageRgba8(image))              } else { -                Err(image_rs::error::ImageError::Limits( -                    image_rs::error::LimitError::from_kind( -                        image_rs::error::LimitErrorKind::DimensionError, +                Err(::image::error::ImageError::Limits( +                    ::image::error::LimitError::from_kind( +                        ::image::error::LimitErrorKind::DimensionError,                      ),                  ))              }          }      }  } - -bitflags! { -    struct Operation: u8 { -        const FLIP_HORIZONTALLY = 0b001; -        const ROTATE_180 = 0b010; -        const FLIP_DIAGONALLY = 0b100; -    } -} - -impl Operation { -    // Meaning of the returned value is described e.g. at: -    // https://magnushoff.com/articles/jpeg-orientation/ -    fn from_exif<R>(reader: &mut R) -> Result<Self, exif::Error> -    where -        R: std::io::BufRead + std::io::Seek, -    { -        let exif = exif::Reader::new().read_from_container(reader)?; - -        Ok(exif -            .get_field(exif::Tag::Orientation, exif::In::PRIMARY) -            .and_then(|field| field.value.get_uint(0)) -            .and_then(|value| u8::try_from(value).ok()) -            .and_then(|value| Self::from_bits(value.saturating_sub(1))) -            .unwrap_or_else(Self::empty)) -    } - -    fn perform(self, mut image: image::DynamicImage) -> image::DynamicImage { -        use image::imageops; - -        if self.contains(Self::FLIP_DIAGONALLY) { -            imageops::flip_vertical_in_place(&mut image); -        } - -        if self.contains(Self::ROTATE_180) { -            imageops::rotate180_in_place(&mut image); -        } - -        if self.contains(Self::FLIP_HORIZONTALLY) { -            imageops::flip_horizontal_in_place(&mut image); -        } - -        image -    } -} diff --git a/graphics/src/layer.rs b/graphics/src/layer.rs new file mode 100644 index 00000000..c9a818fb --- /dev/null +++ b/graphics/src/layer.rs @@ -0,0 +1,144 @@ +//! Draw and stack layers of graphical primitives. +use crate::core::{Rectangle, Transformation}; + +/// A layer of graphical primitives. +/// +/// Layers normally dictate a set of primitives that are +/// rendered in a specific order. +pub trait Layer: Default { +    /// Creates a new [`Layer`] with the given bounds. +    fn with_bounds(bounds: Rectangle) -> Self; + +    /// Flushes and settles any pending group of primitives in the [`Layer`]. +    /// +    /// This will be called when a [`Layer`] is finished. It allows layers to efficiently +    /// record primitives together and defer grouping until the end. +    fn flush(&mut self); + +    /// Resizes the [`Layer`] to the given bounds. +    fn resize(&mut self, bounds: Rectangle); + +    /// Clears all the layers contents and resets its bounds. +    fn reset(&mut self); +} + +/// A stack of layers used for drawing. +#[derive(Debug)] +pub struct Stack<T: Layer> { +    layers: Vec<T>, +    transformations: Vec<Transformation>, +    previous: Vec<usize>, +    current: usize, +    active_count: usize, +} + +impl<T: Layer> Stack<T> { +    /// Creates a new empty [`Stack`]. +    pub fn new() -> Self { +        Self { +            layers: vec![T::default()], +            transformations: vec![Transformation::IDENTITY], +            previous: vec![], +            current: 0, +            active_count: 1, +        } +    } + +    /// Returns a mutable reference to the current [`Layer`] of the [`Stack`], together with +    /// the current [`Transformation`]. +    #[inline] +    pub fn current_mut(&mut self) -> (&mut T, Transformation) { +        let transformation = self.transformation(); + +        (&mut self.layers[self.current], transformation) +    } + +    /// Returns the current [`Transformation`] of the [`Stack`]. +    #[inline] +    pub fn transformation(&self) -> Transformation { +        self.transformations.last().copied().unwrap() +    } + +    /// Pushes a new clipping region in the [`Stack`]; creating a new layer in the +    /// process. +    pub fn push_clip(&mut self, bounds: Rectangle) { +        self.previous.push(self.current); + +        self.current = self.active_count; +        self.active_count += 1; + +        let bounds = bounds * self.transformation(); + +        if self.current == self.layers.len() { +            self.layers.push(T::with_bounds(bounds)); +        } else { +            self.layers[self.current].resize(bounds); +        } +    } + +    /// Pops the current clipping region from the [`Stack`] and restores the previous one. +    /// +    /// The current layer will be recorded for drawing. +    pub fn pop_clip(&mut self) { +        self.flush(); + +        self.current = self.previous.pop().unwrap(); +    } + +    /// Pushes a new [`Transformation`] in the [`Stack`]. +    /// +    /// Future drawing operations will be affected by this new [`Transformation`] until +    /// it is popped using [`pop_transformation`]. +    /// +    /// [`pop_transformation`]: Self::pop_transformation +    pub fn push_transformation(&mut self, transformation: Transformation) { +        self.transformations +            .push(self.transformation() * transformation); +    } + +    /// Pops the current [`Transformation`] in the [`Stack`]. +    pub fn pop_transformation(&mut self) { +        let _ = self.transformations.pop(); +    } + +    /// Returns an iterator over mutable references to the layers in the [`Stack`]. +    pub fn iter_mut(&mut self) -> impl Iterator<Item = &mut T> { +        self.flush(); + +        self.layers[..self.active_count].iter_mut() +    } + +    /// Returns an iterator over immutable references to the layers in the [`Stack`]. +    pub fn iter(&self) -> impl Iterator<Item = &T> { +        self.layers[..self.active_count].iter() +    } + +    /// Returns the slice of layers in the [`Stack`]. +    pub fn as_slice(&self) -> &[T] { +        &self.layers[..self.active_count] +    } + +    /// Flushes and settles any primitives in the current layer of the [`Stack`]. +    pub fn flush(&mut self) { +        self.layers[self.current].flush(); +    } + +    /// Clears the layers of the [`Stack`], allowing reuse. +    /// +    /// This will normally keep layer allocations for future drawing operations. +    pub fn clear(&mut self) { +        for layer in self.layers[..self.active_count].iter_mut() { +            layer.reset(); +        } + +        self.current = 0; +        self.active_count = 1; +        self.previous.clear(); +    } +} + +impl<T: Layer> Default for Stack<T> { +    fn default() -> Self { +        Self::new() +    } +} diff --git a/graphics/src/lib.rs b/graphics/src/lib.rs index d7f2f439..865ebd97 100644 --- a/graphics/src/lib.rs +++ b/graphics/src/lib.rs @@ -7,48 +7,35 @@  #![doc(      html_logo_url = "https://raw.githubusercontent.com/iced-rs/iced/9ab6923e943f784985e9ef9ca28b10278297225d/docs/logo.svg"  )] -#![forbid(rust_2018_idioms)] -#![deny( -    missing_debug_implementations, -    missing_docs, -    unsafe_code, -    unused_results, -    rustdoc::broken_intra_doc_links -)]  #![cfg_attr(docsrs, feature(doc_auto_cfg))]  mod antialiasing;  mod cached; -mod primitive;  mod settings;  mod viewport; -pub mod backend;  pub mod color;  pub mod compositor;  pub mod damage;  pub mod error;  pub mod gradient; +pub mod image; +pub mod layer;  pub mod mesh; -pub mod renderer;  pub mod text;  #[cfg(feature = "geometry")]  pub mod geometry; -#[cfg(feature = "image")] -pub mod image; -  pub use antialiasing::Antialiasing; -pub use backend::Backend;  pub use cached::Cached;  pub use compositor::Compositor; -pub use damage::Damage;  pub use error::Error;  pub use gradient::Gradient; +pub use image::Image; +pub use layer::Layer;  pub use mesh::Mesh; -pub use primitive::Primitive; -pub use renderer::Renderer;  pub use settings::Settings; +pub use text::Text;  pub use viewport::Viewport;  pub use iced_core as core; diff --git a/graphics/src/mesh.rs b/graphics/src/mesh.rs index 20692b07..76602319 100644 --- a/graphics/src/mesh.rs +++ b/graphics/src/mesh.rs @@ -1,8 +1,7 @@  //! Draw triangles!  use crate::color; -use crate::core::{Rectangle, Size}; +use crate::core::{Rectangle, Transformation};  use crate::gradient; -use crate::Damage;  use bytemuck::{Pod, Zeroable}; @@ -14,29 +13,55 @@ pub enum Mesh {          /// The vertices and indices of the mesh.          buffers: Indexed<SolidVertex2D>, -        /// The size of the drawable region of the mesh. -        /// -        /// Any geometry that falls out of this region will be clipped. -        size: Size, +        /// The [`Transformation`] for the vertices of the [`Mesh`]. +        transformation: Transformation, + +        /// The clip bounds of the [`Mesh`]. +        clip_bounds: Rectangle,      },      /// A mesh with a gradient.      Gradient {          /// The vertices and indices of the mesh.          buffers: Indexed<GradientVertex2D>, -        /// The size of the drawable region of the mesh. -        /// -        /// Any geometry that falls out of this region will be clipped. -        size: Size, +        /// The [`Transformation`] for the vertices of the [`Mesh`]. +        transformation: Transformation, + +        /// The clip bounds of the [`Mesh`]. +        clip_bounds: Rectangle,      },  } -impl Damage for Mesh { -    fn bounds(&self) -> Rectangle { +impl Mesh { +    /// Returns the indices of the [`Mesh`]. +    pub fn indices(&self) -> &[u32] { +        match self { +            Self::Solid { buffers, .. } => &buffers.indices, +            Self::Gradient { buffers, .. } => &buffers.indices, +        } +    } + +    /// Returns the [`Transformation`] of the [`Mesh`]. +    pub fn transformation(&self) -> Transformation { +        match self { +            Self::Solid { transformation, .. } +            | Self::Gradient { transformation, .. } => *transformation, +        } +    } + +    /// Returns the clip bounds of the [`Mesh`]. +    pub fn clip_bounds(&self) -> Rectangle {          match self { -            Self::Solid { size, .. } | Self::Gradient { size, .. } => { -                Rectangle::with_size(*size) +            Self::Solid { +                clip_bounds, +                transformation, +                ..              } +            | Self::Gradient { +                clip_bounds, +                transformation, +                .. +            } => *clip_bounds * *transformation,          }      }  } @@ -75,6 +100,47 @@ pub struct GradientVertex2D {      pub gradient: gradient::Packed,  } +/// The result of counting the attributes of a set of meshes. +#[derive(Debug, Clone, Copy, Default)] +pub struct AttributeCount { +    /// The total amount of solid vertices. +    pub solid_vertices: usize, + +    /// The total amount of solid meshes. +    pub solids: usize, + +    /// The total amount of gradient vertices. +    pub gradient_vertices: usize, + +    /// The total amount of gradient meshes. +    pub gradients: usize, + +    /// The total amount of indices. +    pub indices: usize, +} + +/// Returns the number of total vertices & total indices of all [`Mesh`]es. +pub fn attribute_count_of(meshes: &[Mesh]) -> AttributeCount { +    meshes +        .iter() +        .fold(AttributeCount::default(), |mut count, mesh| { +            match mesh { +                Mesh::Solid { buffers, .. } => { +                    count.solids += 1; +                    count.solid_vertices += buffers.vertices.len(); +                    count.indices += buffers.indices.len(); +                } +                Mesh::Gradient { buffers, .. } => { +                    count.gradients += 1; +                    count.gradient_vertices += buffers.vertices.len(); +                    count.indices += buffers.indices.len(); +                } +            } + +            count +        }) +} +  /// A renderer capable of drawing a [`Mesh`].  pub trait Renderer {      /// Draws the given [`Mesh`]. diff --git a/graphics/src/primitive.rs b/graphics/src/primitive.rs deleted file mode 100644 index 6929b0a1..00000000 --- a/graphics/src/primitive.rs +++ /dev/null @@ -1,160 +0,0 @@ -//! Draw using different graphical primitives. -use crate::core::alignment; -use crate::core::image; -use crate::core::svg; -use crate::core::text; -use crate::core::{ -    Background, Border, Color, Font, Pixels, Point, Rectangle, Shadow, -    Transformation, Vector, -}; -use crate::text::editor; -use crate::text::paragraph; - -use std::sync::Arc; - -/// A rendering primitive. -#[derive(Debug, Clone, PartialEq)] -pub enum Primitive<T> { -    /// A text primitive -    Text { -        /// The contents of the text. -        content: String, -        /// The bounds of the text. -        bounds: Rectangle, -        /// The color of the text. -        color: Color, -        /// The size of the text in logical pixels. -        size: Pixels, -        /// The line height of the text. -        line_height: text::LineHeight, -        /// The font of the text. -        font: Font, -        /// The horizontal alignment of the text. -        horizontal_alignment: alignment::Horizontal, -        /// The vertical alignment of the text. -        vertical_alignment: alignment::Vertical, -        /// The shaping strategy of the text. -        shaping: text::Shaping, -        /// The clip bounds of the text. -        clip_bounds: Rectangle, -    }, -    /// A paragraph primitive -    Paragraph { -        /// The [`paragraph::Weak`] reference. -        paragraph: paragraph::Weak, -        /// The position of the paragraph. -        position: Point, -        /// The color of the paragraph. -        color: Color, -        /// The clip bounds of the paragraph. -        clip_bounds: Rectangle, -    }, -    /// An editor primitive -    Editor { -        /// The [`editor::Weak`] reference. -        editor: editor::Weak, -        /// The position of the editor. -        position: Point, -        /// The color of the editor. -        color: Color, -        /// The clip bounds of the editor. -        clip_bounds: Rectangle, -    }, -    /// A raw `cosmic-text` primitive -    RawText(crate::text::Raw), -    /// A quad primitive -    Quad { -        /// The bounds of the quad -        bounds: Rectangle, -        /// The background of the quad -        background: Background, -        /// The [`Border`] of the quad -        border: Border, -        /// The [`Shadow`] of the quad -        shadow: Shadow, -    }, -    /// An image primitive -    Image { -        /// The handle of the image -        handle: image::Handle, -        /// The filter method of the image -        filter_method: image::FilterMethod, -        /// The bounds of the image -        bounds: Rectangle, -    }, -    /// An SVG primitive -    Svg { -        /// The path of the SVG file -        handle: svg::Handle, - -        /// The [`Color`] filter -        color: Option<Color>, - -        /// The bounds of the viewport -        bounds: Rectangle, -    }, -    /// A group of primitives -    Group { -        /// The primitives of the group -        primitives: Vec<Primitive<T>>, -    }, -    /// A clip primitive -    Clip { -        /// The bounds of the clip -        bounds: Rectangle, -        /// The content of the clip -        content: Box<Primitive<T>>, -    }, -    /// A primitive that applies a [`Transformation`] -    Transform { -        /// The [`Transformation`] -        transformation: Transformation, - -        /// The primitive to transform -        content: Box<Primitive<T>>, -    }, -    /// A cached primitive. -    /// -    /// This can be useful if you are implementing a widget where primitive -    /// generation is expensive. -    Cache { -        /// The cached primitive -        content: Arc<Primitive<T>>, -    }, -    /// A backend-specific primitive. -    Custom(T), -} - -impl<T> Primitive<T> { -    /// Groups the current [`Primitive`]. -    pub fn group(primitives: Vec<Self>) -> Self { -        Self::Group { primitives } -    } - -    /// Clips the current [`Primitive`]. -    pub fn clip(self, bounds: Rectangle) -> Self { -        Self::Clip { -            bounds, -            content: Box::new(self), -        } -    } - -    /// Translates the current [`Primitive`]. -    pub fn translate(self, translation: Vector) -> Self { -        Self::Transform { -            transformation: Transformation::translate( -                translation.x, -                translation.y, -            ), -            content: Box::new(self), -        } -    } - -    /// Transforms the current [`Primitive`]. -    pub fn transform(self, transformation: Transformation) -> Self { -        Self::Transform { -            transformation, -            content: Box::new(self), -        } -    } -} diff --git a/graphics/src/renderer.rs b/graphics/src/renderer.rs deleted file mode 100644 index f517ff3e..00000000 --- a/graphics/src/renderer.rs +++ /dev/null @@ -1,269 +0,0 @@ -//! Create a renderer from a [`Backend`]. -use crate::backend::{self, Backend}; -use crate::compositor; -use crate::core; -use crate::core::image; -use crate::core::renderer; -use crate::core::svg; -use crate::core::text::Text; -use crate::core::{ -    Background, Color, Font, Pixels, Point, Rectangle, Size, Transformation, -}; -use crate::mesh; -use crate::text; -use crate::{Mesh, Primitive}; - -use std::borrow::Cow; - -/// A backend-agnostic renderer that supports all the built-in widgets. -#[derive(Debug)] -pub struct Renderer<B: Backend> { -    backend: B, -    default_font: Font, -    default_text_size: Pixels, -    primitives: Vec<Primitive<B::Primitive>>, -    stack: Vec<Vec<Primitive<B::Primitive>>>, -} - -impl<B: Backend> Renderer<B> { -    /// Creates a new [`Renderer`] from the given [`Backend`]. -    pub fn new( -        backend: B, -        default_font: Font, -        default_text_size: Pixels, -    ) -> Self { -        Self { -            backend, -            default_font, -            default_text_size, -            primitives: Vec::new(), -            stack: Vec::new(), -        } -    } - -    /// Returns a reference to the [`Backend`] of the [`Renderer`]. -    pub fn backend(&self) -> &B { -        &self.backend -    } - -    /// Enqueues the given [`Primitive`] in the [`Renderer`] for drawing. -    pub fn draw_primitive(&mut self, primitive: Primitive<B::Primitive>) { -        self.primitives.push(primitive); -    } - -    /// Runs the given closure with the [`Backend`] and the recorded primitives -    /// of the [`Renderer`]. -    pub fn with_primitives<O>( -        &mut self, -        f: impl FnOnce(&mut B, &[Primitive<B::Primitive>]) -> O, -    ) -> O { -        f(&mut self.backend, &self.primitives) -    } -} - -impl<B: Backend> iced_core::Renderer for Renderer<B> { -    fn start_layer(&mut self) { -        self.stack.push(std::mem::take(&mut self.primitives)); -    } - -    fn end_layer(&mut self, bounds: Rectangle) { -        let layer = std::mem::replace( -            &mut self.primitives, -            self.stack.pop().expect("a layer should be recording"), -        ); - -        self.primitives.push(Primitive::group(layer).clip(bounds)); -    } - -    fn start_transformation(&mut self) { -        self.stack.push(std::mem::take(&mut self.primitives)); -    } - -    fn end_transformation(&mut self, transformation: Transformation) { -        let layer = std::mem::replace( -            &mut self.primitives, -            self.stack.pop().expect("a layer should be recording"), -        ); - -        self.primitives -            .push(Primitive::group(layer).transform(transformation)); -    } - -    fn fill_quad( -        &mut self, -        quad: renderer::Quad, -        background: impl Into<Background>, -    ) { -        self.primitives.push(Primitive::Quad { -            bounds: quad.bounds, -            background: background.into(), -            border: quad.border, -            shadow: quad.shadow, -        }); -    } - -    fn clear(&mut self) { -        self.primitives.clear(); -    } -} - -impl<B> core::text::Renderer for Renderer<B> -where -    B: Backend + backend::Text, -{ -    type Font = Font; -    type Paragraph = text::Paragraph; -    type Editor = text::Editor; - -    const ICON_FONT: Font = Font::with_name("Iced-Icons"); -    const CHECKMARK_ICON: char = '\u{f00c}'; -    const ARROW_DOWN_ICON: char = '\u{e800}'; - -    fn default_font(&self) -> Self::Font { -        self.default_font -    } - -    fn default_size(&self) -> Pixels { -        self.default_text_size -    } - -    fn load_font(&mut self, bytes: Cow<'static, [u8]>) { -        self.backend.load_font(bytes); -    } - -    fn fill_paragraph( -        &mut self, -        paragraph: &Self::Paragraph, -        position: Point, -        color: Color, -        clip_bounds: Rectangle, -    ) { -        self.primitives.push(Primitive::Paragraph { -            paragraph: paragraph.downgrade(), -            position, -            color, -            clip_bounds, -        }); -    } - -    fn fill_editor( -        &mut self, -        editor: &Self::Editor, -        position: Point, -        color: Color, -        clip_bounds: Rectangle, -    ) { -        self.primitives.push(Primitive::Editor { -            editor: editor.downgrade(), -            position, -            color, -            clip_bounds, -        }); -    } - -    fn fill_text( -        &mut self, -        text: Text<'_, Self::Font>, -        position: Point, -        color: Color, -        clip_bounds: Rectangle, -    ) { -        self.primitives.push(Primitive::Text { -            content: text.content.to_string(), -            bounds: Rectangle::new(position, text.bounds), -            size: text.size, -            line_height: text.line_height, -            color, -            font: text.font, -            horizontal_alignment: text.horizontal_alignment, -            vertical_alignment: text.vertical_alignment, -            shaping: text.shaping, -            clip_bounds, -        }); -    } -} - -impl<B> image::Renderer for Renderer<B> -where -    B: Backend + backend::Image, -{ -    type Handle = image::Handle; - -    fn measure_image(&self, handle: &image::Handle) -> Size<u32> { -        self.backend().dimensions(handle) -    } - -    fn draw_image( -        &mut self, -        handle: image::Handle, -        filter_method: image::FilterMethod, -        bounds: Rectangle, -    ) { -        self.primitives.push(Primitive::Image { -            handle, -            filter_method, -            bounds, -        }); -    } -} - -impl<B> svg::Renderer for Renderer<B> -where -    B: Backend + backend::Svg, -{ -    fn measure_svg(&self, handle: &svg::Handle) -> Size<u32> { -        self.backend().viewport_dimensions(handle) -    } - -    fn draw_svg( -        &mut self, -        handle: svg::Handle, -        color: Option<Color>, -        bounds: Rectangle, -    ) { -        self.primitives.push(Primitive::Svg { -            handle, -            color, -            bounds, -        }); -    } -} - -impl<B: Backend> mesh::Renderer for Renderer<B> { -    fn draw_mesh(&mut self, mesh: Mesh) { -        match B::Primitive::try_from(mesh) { -            Ok(primitive) => { -                self.draw_primitive(Primitive::Custom(primitive)); -            } -            Err(error) => { -                log::warn!("mesh primitive could not be drawn: {error:?}"); -            } -        } -    } -} - -#[cfg(feature = "geometry")] -impl<B> crate::geometry::Renderer for Renderer<B> -where -    B: Backend + crate::geometry::Backend, -    B::Frame: -        crate::geometry::frame::Backend<Geometry = Primitive<B::Primitive>>, -{ -    type Frame = B::Frame; -    type Geometry = Primitive<B::Primitive>; - -    fn new_frame(&self, size: Size) -> Self::Frame { -        self.backend.new_frame(size) -    } - -    fn draw_geometry(&mut self, geometry: Self::Geometry) { -        self.draw_primitive(geometry); -    } -} - -impl<B> compositor::Default for Renderer<B> -where -    B: Backend, -{ -    type Compositor = B::Compositor; -} diff --git a/graphics/src/settings.rs b/graphics/src/settings.rs index 68673536..2e8275c6 100644 --- a/graphics/src/settings.rs +++ b/graphics/src/settings.rs @@ -1,7 +1,7 @@  use crate::core::{Font, Pixels};  use crate::Antialiasing; -/// The settings of a Backend. +/// The settings of a renderer.  #[derive(Debug, Clone, Copy, PartialEq)]  pub struct Settings {      /// The default [`Font`] to use. diff --git a/graphics/src/text.rs b/graphics/src/text.rs index 0310ead7..30269e69 100644 --- a/graphics/src/text.rs +++ b/graphics/src/text.rs @@ -9,14 +9,141 @@ pub use paragraph::Paragraph;  pub use cosmic_text; +use crate::core::alignment;  use crate::core::font::{self, Font};  use crate::core::text::Shaping; -use crate::core::{Color, Point, Rectangle, Size}; +use crate::core::{Color, Pixels, Point, Rectangle, Size, Transformation};  use once_cell::sync::OnceCell;  use std::borrow::Cow;  use std::sync::{Arc, RwLock, Weak}; +/// A text primitive. +#[derive(Debug, Clone, PartialEq)] +pub enum Text { +    /// A paragraph. +    #[allow(missing_docs)] +    Paragraph { +        paragraph: paragraph::Weak, +        position: Point, +        color: Color, +        clip_bounds: Rectangle, +        transformation: Transformation, +    }, +    /// An editor. +    #[allow(missing_docs)] +    Editor { +        editor: editor::Weak, +        position: Point, +        color: Color, +        clip_bounds: Rectangle, +        transformation: Transformation, +    }, +    /// Some cached text. +    Cached { +        /// The contents of the text. +        content: String, +        /// The bounds of the text. +        bounds: Rectangle, +        /// The color of the text. +        color: Color, +        /// The size of the text in logical pixels. +        size: Pixels, +        /// The line height of the text. +        line_height: Pixels, +        /// The font of the text. +        font: Font, +        /// The horizontal alignment of the text. +        horizontal_alignment: alignment::Horizontal, +        /// The vertical alignment of the text. +        vertical_alignment: alignment::Vertical, +        /// The shaping strategy of the text. +        shaping: Shaping, +        /// The clip bounds of the text. +        clip_bounds: Rectangle, +    }, +    /// Some raw text. +    #[allow(missing_docs)] +    Raw { +        raw: Raw, +        transformation: Transformation, +    }, +} + +impl Text { +    /// Returns the visible bounds of the [`Text`]. +    pub fn visible_bounds(&self) -> Option<Rectangle> { +        let (bounds, horizontal_alignment, vertical_alignment) = match self { +            Text::Paragraph { +                position, +                paragraph, +                clip_bounds, +                transformation, +                .. +            } => ( +                Rectangle::new(*position, paragraph.min_bounds) +                    .intersection(clip_bounds) +                    .map(|bounds| bounds * *transformation), +                Some(paragraph.horizontal_alignment), +                Some(paragraph.vertical_alignment), +            ), +            Text::Editor { +                editor, +                position, +                clip_bounds, +                transformation, +                .. +            } => ( +                Rectangle::new(*position, editor.bounds) +                    .intersection(clip_bounds) +                    .map(|bounds| bounds * *transformation), +                None, +                None, +            ), +            Text::Cached { +                bounds, +                clip_bounds, +                horizontal_alignment, +                vertical_alignment, +                .. +            } => ( +                bounds.intersection(clip_bounds), +                Some(*horizontal_alignment), +                Some(*vertical_alignment), +            ), +            Text::Raw { raw, .. } => (Some(raw.clip_bounds), None, None), +        }; + +        let mut bounds = bounds?; + +        if let Some(alignment) = horizontal_alignment { +            match alignment { +                alignment::Horizontal::Left => {} +                alignment::Horizontal::Center => { +                    bounds.x -= bounds.width / 2.0; +                } +                alignment::Horizontal::Right => { +                    bounds.x -= bounds.width; +                } +            } +        } + +        if let Some(alignment) = vertical_alignment { +            match alignment { +                alignment::Vertical::Top => {} +                alignment::Vertical::Center => { +                    bounds.y -= bounds.height / 2.0; +                } +                alignment::Vertical::Bottom => { +                    bounds.y -= bounds.height; +                } +            } +        } + +        Some(bounds) +    } +} +  /// The regular variant of the [Fira Sans] font.  ///  /// It is loaded as part of the default fonts in Wasm builds. diff --git a/graphics/src/text/cache.rs b/graphics/src/text/cache.rs index 7fb33567..822b61c4 100644 --- a/graphics/src/text/cache.rs +++ b/graphics/src/text/cache.rs @@ -2,22 +2,18 @@  use crate::core::{Font, Size};  use crate::text; -use rustc_hash::{FxHashMap, FxHashSet}; +use rustc_hash::{FxHashMap, FxHashSet, FxHasher};  use std::collections::hash_map; -use std::hash::{BuildHasher, Hash, Hasher}; +use std::hash::{Hash, Hasher};  /// A store of recently used sections of text. -#[allow(missing_debug_implementations)] -#[derive(Default)] +#[derive(Debug, Default)]  pub struct Cache {      entries: FxHashMap<KeyHash, Entry>,      aliases: FxHashMap<KeyHash, KeyHash>,      recently_used: FxHashSet<KeyHash>, -    hasher: HashBuilder,  } -type HashBuilder = xxhash_rust::xxh3::Xxh3Builder; -  impl Cache {      /// Creates a new empty [`Cache`].      pub fn new() -> Self { @@ -35,7 +31,7 @@ impl Cache {          font_system: &mut cosmic_text::FontSystem,          key: Key<'_>,      ) -> (KeyHash, &mut Entry) { -        let hash = key.hash(self.hasher.build_hasher()); +        let hash = key.hash(FxHasher::default());          if let Some(hash) = self.aliases.get(&hash) {              let _ = self.recently_used.insert(*hash); @@ -77,7 +73,7 @@ impl Cache {              ] {                  if key.bounds != bounds {                      let _ = self.aliases.insert( -                        Key { bounds, ..key }.hash(self.hasher.build_hasher()), +                        Key { bounds, ..key }.hash(FxHasher::default()),                          hash,                      );                  } @@ -138,7 +134,7 @@ impl Key<'_> {  pub type KeyHash = u64;  /// A cache entry. -#[allow(missing_debug_implementations)] +#[derive(Debug)]  pub struct Entry {      /// The buffer of text, ready for drawing.      pub buffer: cosmic_text::Buffer, diff --git a/graphics/src/text/paragraph.rs b/graphics/src/text/paragraph.rs index 5d027542..31a323ac 100644 --- a/graphics/src/text/paragraph.rs +++ b/graphics/src/text/paragraph.rs @@ -61,7 +61,7 @@ impl Paragraph {  impl core::text::Paragraph for Paragraph {      type Font = Font; -    fn with_text(text: Text<'_, Font>) -> Self { +    fn with_text(text: Text<&str>) -> Self {          log::trace!("Allocating paragraph: {}", text.content);          let mut font_system = @@ -146,7 +146,7 @@ impl core::text::Paragraph for Paragraph {          }      } -    fn compare(&self, text: Text<'_, Font>) -> core::text::Difference { +    fn compare(&self, text: Text<&str>) -> core::text::Difference {          let font_system = text::font_system().read().expect("Read font system");          let paragraph = self.internal();          let metrics = paragraph.buffer.metrics(); | 
