diff --git a/cairo/src/context.rs b/cairo/src/context.rs index 0400386f8a9d..4869ec608c68 100644 --- a/cairo/src/context.rs +++ b/cairo/src/context.rs @@ -51,6 +51,46 @@ impl fmt::Debug for RectangleList { } } +/// A drawing context, used to draw on [`Surface`]s. +/// +/// [`Context`] is the main entry point for all drawing operations. To acquire a [`Context`], you +/// can create any kind of [`Surface`] (such as [`ImageSurface`] or [`PdfSurface`]), and call +/// [`Context::new`] with the target surface: +/// +/// ``` +/// use cairo::{Context, Format, ImageSurface}; +/// +/// fn main() -> Result<(), cairo::Error> { +/// let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; +/// let ctx = Context::new(&surface)?; +/// +/// // paint the background black +/// ctx.set_source_rgb(0.0, 0.0, 0.0); +/// ctx.paint(); +/// +/// // etc., other drawing operations +/// Ok(()) +/// } +/// ``` +/// +/// Each example in the documentation below can be used with the above setup. To view the images +/// generated by the examples, you can render the surface to a file. For example, to render the +/// surface to a PNG file (requires the `png` feature to be enabled), you can use the following +/// code after drawing on the surface: +/// +/// ``` +/// # fn main() -> Result<(), cairo::Error> { +/// # use cairo::{Context, Format, ImageSurface}; +/// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; +/// # let ctx = Context::new(&surface)?; +/// let mut file = std::fs::File::create("example.png")?; +/// surface.write_to_png(&mut file)?; +/// # Ok(()) +/// # } +/// ``` +/// +/// [`ImageSurface`]: crate::ImageSurface +/// [`PdfSurface`]: crate::PdfSurface #[derive(Debug)] #[repr(transparent)] #[doc(alias = "cairo_t")] @@ -156,6 +196,9 @@ impl Context { self.0.as_ptr() } + /// Checks whether an error has previously occurred for this context. + /// + /// Refer to [`Error`] for an exhaustive list of error codes. #[doc(alias = "cairo_status")] #[inline] pub fn status(&self) -> Result<(), Error> { @@ -163,67 +206,253 @@ impl Context { status_to_result(status) } + /// Creates a new [`Context`] with default graphics state parameters, for drawing to a target + /// [`Surface`]. pub fn new(target: impl AsRef) -> Result { let ctx = unsafe { Self::from_raw_full(ffi::cairo_create(target.as_ref().to_raw_none())) }; ctx.status().map(|_| ctx) } + /// Makes a copy of the current state of the [`Context`], saving it in an internal stack of + /// saved states for the [`Context`]. When [`Context::restore`] is called, this context will be + /// restored to the saved state. + /// + /// Multiple calls to [`Context::save`] and [`Context::restore`] can be nested; each call to + /// [`Context::restore`] restores the state from the matching paired [`Context::save`]. + /// + /// # Examples + /// + /// ``` + /// # fn main() -> Result<(), cairo::Error> { + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; + /// // set source to red + /// ctx.set_source_rgb(1.0, 0.0, 0.0); + /// + /// // this block scope is not necessary, but can help with visualizing what happens + /// { + /// // save current state + /// ctx.save()?; + /// + /// // set source to white + /// ctx.set_source_rgb(1.0, 1.0, 1.0); + /// + /// // shift origin in canvas space, to (50, 50) + /// ctx.translate(50.0, 50.0); + /// + /// // draw a white square with its upper-left corner at (0, 0) in canvas space + /// // (i.e. at (50, 50) in device space) + /// ctx.rectangle(0.0, 0.0, 20.0, 20.0); + /// ctx.fill()?; + /// + /// // restore state, origin back at (0, 0) in canvas space + /// ctx.restore()?; + /// } + /// + /// // draw a red square at (0, 0) in canvas and device space + /// ctx.rectangle(0.0, 0.0, 20.0, 20.0); + /// ctx.fill()?; + /// # Ok(()) + /// # } + /// ``` #[doc(alias = "cairo_save")] pub fn save(&self) -> Result<(), Error> { unsafe { ffi::cairo_save(self.0.as_ptr()) } self.status() } + /// Restores the [`Context`] to the state saved by a preceding call to [`Context::save`], + /// removing that state from the stack of saved states. + /// + /// See [`Context::save`] for example usage. #[doc(alias = "cairo_restore")] pub fn restore(&self) -> Result<(), Error> { unsafe { ffi::cairo_restore(self.0.as_ptr()) } self.status() } + /// Returns the target [`Surface`] for this [`Context`]. #[doc(alias = "get_target")] #[doc(alias = "cairo_get_target")] pub fn target(&self) -> Surface { unsafe { Surface::from_raw_none(ffi::cairo_get_target(self.0.as_ptr())) } } + /// Temporarily redirects drawing to an intermediate surface, known as a group, with a default + /// content type of [`Content::ColorAlpha`]. The redirection lasts until the group is completed + /// by a call to [`Context::pop_group`] / [`Context::pop_group_to_source`]. + /// + /// This can be convenient for performing intermediate compositing. A common use of a group is + /// to render objects as opaque within the group (so that they occlude each other), and then + /// blend the result with translucence onto the destination. + /// + /// Groups can be nested arbitrarily deep by making balanced calls to [`Context::push_group`] / + /// [`Context::pop_group`]. Each call pushes / pops the new target group onto / from a stack. + /// + /// This function calls [`Context::save`] so that any changes to the graphics state will not be + /// visible outside the group. ([`Context::pop_group`] functions call [`Context::restore`]) + /// + /// # Examples + /// + /// TODO: make a nice example here #[doc(alias = "cairo_push_group")] pub fn push_group(&self) { unsafe { ffi::cairo_push_group(self.0.as_ptr()) } } + /// Temporarily redirects drawing to an intermediate surface, known as a group, with the given + /// content type. The redirection lasts until the group is completed by a call to + /// [`Context::pop_group`] / [`Context::pop_group_to_source`]. + /// + /// See [`Context::push_group`] for more details. #[doc(alias = "cairo_push_group_with_content")] pub fn push_group_with_content(&self, content: Content) { unsafe { ffi::cairo_push_group_with_content(self.0.as_ptr(), content.into()) } } + /// Terminates the redirection begun by a call to [`Context::push_group`] or + /// [`Context::push_group_with_content`] and returns a new [`Pattern`] containing the results + /// of all drawing operations performed on the group. + /// + /// This function calls [`Context::restore`] so that any changes to the graphics state will not + /// be visible outside the group. + /// + /// # Examples + /// + /// TODO: make a nice example here #[doc(alias = "cairo_pop_group")] pub fn pop_group(&self) -> Result { let pattern = unsafe { Pattern::from_raw_full(ffi::cairo_pop_group(self.0.as_ptr())) }; self.status().map(|_| pattern) } + /// Terminates the redirection begun by a call to [`Context::push_group`] or + /// [`Context::push_group_with_content`]. The results of the drawing operations are then used + /// as the source pattern for this [`Context`]. + /// + /// This function calls [`Context::restore`] so that any changes to the graphics state will not + /// be visible outside the group. + /// + /// # Examples + /// + /// TODO: make a nice example here #[doc(alias = "cairo_pop_group_to_source")] pub fn pop_group_to_source(&self) -> Result<(), Error> { unsafe { ffi::cairo_pop_group_to_source(self.0.as_ptr()) }; self.status() } + /// Gets the current destination surface for the [`Context`]. + /// + /// This is either the original target surface as passed to [`Context::new`], or the target + /// surface for the current group as started by the most recent call to [`Context::push_group`] + /// / [`Context::push_group_with_content`]. #[doc(alias = "get_group_target")] #[doc(alias = "cairo_get_group_target")] pub fn group_target(&self) -> Surface { unsafe { Surface::from_raw_none(ffi::cairo_get_group_target(self.0.as_ptr())) } } + /// Sets this [`Context`]'s source pattern to an opaque color. This opaque color will be used + /// for any subsequent drawing operation. + /// + /// The color components are floating point numbers in the range `0.0..=1.0`. If the given + /// values are outside that range, they will be clamped. + /// + /// The default source pattern is an opaque black `(0.0, 0.0, 0.0)`. + /// + /// # Examples + /// + /// ``` + /// # fn main() -> Result<(), cairo::Error> { + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; + /// // paint the background black + /// ctx.paint()?; + /// + /// // draw a red circle centered at (50, 50) + /// ctx.set_source_rgb(1.0, 0.0, 0.0); + /// ctx.arc(50.0, 50.0, 20.0, 0.0, std::f64::consts::TAU); + /// ctx.fill()?; + /// # Ok(()) + /// # } + /// ``` #[doc(alias = "cairo_set_source_rgb")] pub fn set_source_rgb(&self, red: f64, green: f64, blue: f64) { unsafe { ffi::cairo_set_source_rgb(self.0.as_ptr(), red, green, blue) } } + /// Sets this [`Context`]'s source pattern to a translucent color. This translucent color will + /// be used for any subsequent drawing operation. + /// + /// The color and alpha components are floating point numbers in the range `0.0..=1.0`. If the + /// given values are outside that range, they will be clamped. + /// + /// The default source pattern is an opaque black `(0.0, 0.0, 0.0, 1.0)`. + /// + /// # Examples + /// + /// ``` + /// # fn main() -> Result<(), cairo::Error> { + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; + /// // paint the background black + /// ctx.paint()?; + /// + /// // draw a red circle centered at (50, 50) + /// ctx.set_source_rgb(1.0, 0.0, 0.0); + /// ctx.arc(50.0, 50.0, 20.0, 0.0, std::f64::consts::TAU); + /// ctx.fill()?; + /// + /// // draw a translucent blue square on top of the circle + /// ctx.set_source_rgba(0.0, 0.0, 1.0, 0.5); + /// ctx.rectangle(30.0, 30.0, 40.0, 40.0); + /// ctx.fill()?; + /// # Ok(()) + /// # } + /// ``` #[doc(alias = "cairo_set_source_rgba")] pub fn set_source_rgba(&self, red: f64, green: f64, blue: f64, alpha: f64) { unsafe { ffi::cairo_set_source_rgba(self.0.as_ptr(), red, green, blue, alpha) } } + /// Sets this [`Context`]'s source pattern. This source pattern will be used for any subsequent + /// drawing operation. + /// + /// **Note**: The pattern's transformation matrix will be locked to the user space when + /// [`Context::set_source`] is called. Any subsequent modification of the current transformation + /// matrix **will not** affect the source pattern. + /// + /// The default source pattern is an opaque black `(0.0, 0.0, 0.0, 1.0)`. + /// + /// # Examples + /// + /// ``` + /// # fn main() -> Result<(), cairo::Error> { + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; + /// // create a radial gradient from red to blue + /// + /// // gradient will begin at (50, 50) with radius 0 + /// // ends at (50, 50) with radius 50 + /// let pattern = RadialGradient::new(50.0, 50.0, 0.0, 50.0, 50.0, 50.0); + /// + /// // first color stop: at offset=0, draw with red=(1, 0, 0) + /// pattern.add_color_stop_rgb(0.0, 1.0, 0.0, 0.0); + /// + /// // second color stop: at offset=1, draw with blue=(0, 0, 1) + /// pattern.add_color_stop_rgb(1.0, 0.0, 0.0, 1.0); + /// + /// // paint the canvas with the gradient + /// ctx.set_source(&pattern)?; + /// ctx.paint()?; + /// # Ok(()) + /// # } + /// ``` #[doc(alias = "cairo_set_source")] pub fn set_source(&self, source: impl AsRef) -> Result<(), Error> { let source = source.as_ref(); @@ -234,6 +463,7 @@ impl Context { self.status() } + /// Returns the current source pattern for this [`Context`]. #[doc(alias = "get_source")] #[doc(alias = "cairo_get_source")] pub fn source(&self) -> Pattern { @@ -502,6 +732,7 @@ impl Context { self.status() } + /// Paints the current source color everywhere within the current clip region. #[doc(alias = "cairo_paint")] pub fn paint(&self) -> Result<(), Error> { unsafe { ffi::cairo_paint(self.0.as_ptr()) }; @@ -844,6 +1075,71 @@ impl Context { unsafe { ffi::cairo_new_sub_path(self.0.as_ptr()) } } + /// Adds a line segment from the current point to the beginning of the current sub-path (the + /// most recent point passed to [`Context::move_to`]) to the path, and closes this sub-path. + /// After this call, the current point will be at the **joined endpoint of the sub-path**. + /// + /// The behavior of [`Context::close_path`] is distinct from simply calling + /// [`Context::line_to`] with the equivalent coordinate in the case of stroking. When a closed + /// sub-path is stroked, there are no caps on the ends of the sub-path. Instead, there is a + /// line join connecting the final and initial segments of the sub-path. + /// + /// If there is no current point before the call to [`Context::close_path`], this function will + /// have no effect. + /// + /// This function maps to [`PathSegment::ClosePath`]. + /// + /// **Note**: Calls to [`Context::close_path`] now (as of `cairo 1.2.4`) place an explicit + /// [`MoveTo`] element into the path immediately after the [`ClosePath`] element. This can + /// simplify path processing in some cases. See below for an example of this case. + /// + /// # Examples + /// + /// ``` + /// # fn main() -> Result<(), cairo::Error> { + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; + /// // paint the background black + /// ctx.paint()?; + /// + /// // draw a 5-pointed star with a white fill, in similar manner to how a human would + /// let angle = std::f64::consts::TAU / 2.5; + /// ctx.set_source_rgb(1.0, 1.0, 1.0); + /// for i in 0..5 { + /// let x = 50.0 + 40.0 * (i as f64 * angle).cos(); + /// let y = 50.0 + 40.0 * (i as f64 * angle).sin(); + /// if i == 0 { + /// ctx.move_to(x, y); + /// } else { + /// ctx.line_to(x, y); + /// } + /// } + /// ctx.close_path(); + /// + /// // print the path details before filling (will clear the context path) + /// let path = ctx.copy_path()?; + /// for element in path.iter() { + /// println!("{:?}", element); + /// } + /// + /// ctx.fill()?; + /// + /// // output: + /// // MoveTo((90.0, 50.0)) + /// // LineTo((17.640625, 73.51171875)) + /// // LineTo((62.359375, 11.95703125)) + /// // LineTo((62.359375, 88.04296875)) + /// // LineTo((17.640625, 26.48828125)) + /// // ClosePath + /// // MoveTo((90.0, 50.0)) + /// # Ok(()) + /// # } + /// ``` + /// + /// [`PathSegment::ClosePath`]: crate::PathSegment::ClosePath + /// [`MoveTo`]: crate::PathSegment::MoveTo + /// [`ClosePath`]: crate::PathSegment::ClosePath #[doc(alias = "cairo_close_path")] pub fn close_path(&self) { unsafe { ffi::cairo_close_path(self.0.as_ptr()) } @@ -859,16 +1155,102 @@ impl Context { unsafe { ffi::cairo_arc_negative(self.0.as_ptr(), xc, yc, radius, angle1, angle2) } } + /// Adds a cubic Bézier spline from the current point to position `(x3, y3)` in user-space + /// coordinates to the path, using `(x1, y1)` and `(x2, y2)` as the control points. After this + /// call, the current point will be `(x3, y3)`. + /// + /// If there is no current point before the call to [`Context::curve_to`], this function will + /// behave as if preceded by a call to [`Context::move_to(&ctx, x1, y1)`]. + /// + /// This function maps to [`PathSegment::CurveTo`]. + /// + /// # Examples + /// + /// ``` + /// # fn main() -> Result<(), cairo::Error> { + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; + /// // paint the background black + /// ctx.paint()?; + /// + /// // draw an infinity symbol with a white stroke like how a human would + /// ctx.set_source_rgb(1.0, 1.0, 1.0); + /// ctx.move_to(20.0, 50.0); + /// ctx.curve_to( // left loop, going up + /// 20.0, 33.0, + /// 40.0, 33.0, + /// 50.0, 50.0, + /// ); + /// ctx.curve_to( // right loop, going down + /// 60.0, 67.0, + /// 80.0, 67.0, + /// 80.0, 50.0, + /// ); + /// ctx.curve_to( // right loop, going up + /// 80.0, 33.0, + /// 60.0, 33.0, + /// 50.0, 50.0, + /// ); + /// ctx.curve_to( // left loop, going down + /// 40.0, 67.0, + /// 20.0, 67.0, + /// 20.0, 50.0, + /// ); + /// + /// // print the path details before stroking (will clear the path) + /// let path = ctx.copy_path()?; + /// for element in path.iter() { + /// println!("{:?}", element); + /// } + /// + /// ctx.stroke()?; + /// + /// // output: + /// // MoveTo((20.0, 50.0)) + /// // CurveTo((20.0, 33.0), (40.0, 33.0), (50.0, 50.0)) + /// // CurveTo((60.0, 67.0), (80.0, 67.0), (80.0, 50.0)) + /// // CurveTo((80.0, 33.0), (60.0, 33.0), (50.0, 50.0)) + /// // CurveTo((40.0, 67.0), (20.0, 67.0), (20.0, 50.0)) + /// + /// # Ok(()) + /// # } + /// ``` + /// + /// [`Context::move_to(&ctx, x1, y1)`]: crate::Context::move_to + /// [`PathSegment::CurveTo`]: crate::PathSegment::CurveTo #[doc(alias = "cairo_curve_to")] pub fn curve_to(&self, x1: f64, y1: f64, x2: f64, y2: f64, x3: f64, y3: f64) { unsafe { ffi::cairo_curve_to(self.0.as_ptr(), x1, y1, x2, y2, x3, y3) } } + /// Adds a line segment from the current point to the given point `(x, y)`, to the path. After + /// this call, the current point will be `(x, y)`. + /// + /// If there is no current point before the call to [`Context::line_to`], this function will + /// behave as [`Context::move_to`]. + /// + /// This function maps to [`PathSegment::LineTo`]. + /// + /// # Examples + /// + /// See [`Context::close_path`] for an example of using [`Context::line_to`]. + /// + /// [`PathSegment::LineTo`]: crate::PathSegment::LineTo #[doc(alias = "cairo_line_to")] pub fn line_to(&self, x: f64, y: f64) { unsafe { ffi::cairo_line_to(self.0.as_ptr(), x, y) } } + /// Begins a new sub-path. After this call, the current point will be `(x, y)`. + /// + /// This function maps to [`PathSegment::MoveTo`]. + /// + /// # Examples + /// + /// See [`Context::close_path`] for an example of using [`Context::move_to`]. + /// + /// [`PathSegment::MoveTo`]: crate::PathSegment::MoveTo #[doc(alias = "cairo_move_to")] pub fn move_to(&self, x: f64, y: f64) { unsafe { ffi::cairo_move_to(self.0.as_ptr(), x, y) } diff --git a/cairo/src/enums.rs b/cairo/src/enums.rs index 091ff89fb854..91560cba74eb 100644 --- a/cairo/src/enums.rs +++ b/cairo/src/enums.rs @@ -53,28 +53,58 @@ macro_rules! gvalue_impl { }; } +/// Specifies the type of antialiasing to do when rendering text or shapes. +/// +/// `cairo 1.12` added a set of antialiasing hints, rather than specifying a specific antialias +/// method. These hints are: +/// +/// - [`Antialias::Fast`]: Allow the backend to degrade raster quality for speed. +/// - [`Antialias::Good`]: Balance between speed and quality. +/// - [`Antialias::Best`]: High-fidelity, but potentially slow, raster mode. +/// +/// These make no guarantee on how the backend will perform its rasterisation (if it even +/// rasterises!), nor that they have any differing effect other than to enable some form of +/// antialiasing. In the case of glyph rendering, [`Antialias::Fast`] and [`Antialias::Good`] will +/// be mapped to [`Antialias::Gray`], with [`Antialias::Best`] being equivalent to +/// [`Antialias::Subpixel`]. +/// +/// The interpretation of [`Antialias::Default`] is left entirely up to the backend. Typically, +/// this will be similar to [`Antialias::Good`]. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_antialias_t")] pub enum Antialias { + /// Use the default antialiasing for the subsystem and target device. #[doc(alias = "ANTIALIAS_DEFAULT")] Default, - /* method */ + /// Use a bilevel alpha mask. #[doc(alias = "ANTIALIAS_NONE")] None, + + /// Perform single-color antialiasing (using shades of gray for black text on a white + /// background, for example). #[doc(alias = "ANTIALIAS_GRAY")] Gray, + + /// Perform antialiasing by taking advantage of the order of subpixel elements on devices such + /// as LCD panels. #[doc(alias = "ANTIALIAS_SUBPIXEL")] Subpixel, /* hints */ + /// Hint that the backend should perform some antialiasing, but prefer speed over quality. #[doc(alias = "ANTIALIAS_FAST")] Fast, + + /// Hint that the backend should balance quality against performance. #[doc(alias = "ANTIALIAS_GOOD")] Good, + + /// Hint that the backend should render at the highest quality, sacrificing speed if necessary. #[doc(alias = "ANTIALIAS_BEST")] Best, + #[doc(hidden)] __Unknown(i32), } @@ -114,14 +144,32 @@ impl From for Antialias { #[cfg(feature = "use_glib")] gvalue_impl!(Antialias, ffi::gobject::cairo_gobject_antialias_get_type); +/// Select how paths are filled. +/// +/// For both fill rules, whether or not a point is included in the fill is determined by taking a +/// ray from that point to infinity, and looking at intersections with the path. The ray can be in +/// any direction, as long as it doesn't pass through the end point of a segment, or have a tricky +/// intersection, such as intersecting tangent to the path. +/// +/// The default fill rule is [`FillRule::Winding`]. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_fill_rule_t")] pub enum FillRule { + /// Count the number of times the ray crosses the path from left to right. + /// + /// When crossing a path from left to right, increment the count by 1. When crossing from right + /// to left, decrement the count by 1. If the result is non-zero, the point will be filled. #[doc(alias = "FILL_RULE_WINDING")] Winding, + + /// Count the number of intersections of the ray with the path, with no regard to the + /// orientation of the crossing. + /// + /// If the total number of intersections is odd, the point will be filled. #[doc(alias = "FILL_RULE_EVEN_ODD")] EvenOdd, + #[doc(hidden)] __Unknown(i32), } @@ -151,16 +199,25 @@ impl From for FillRule { #[cfg(feature = "use_glib")] gvalue_impl!(FillRule, ffi::gobject::cairo_gobject_fill_rule_get_type); +/// Specifies how to render endpoints of paths when stroking. +/// +/// The default line cap style is [`LineCap::Butt`]. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_line_cap_t")] pub enum LineCap { + /// Begin and end the line exactly at the start and end points. #[doc(alias = "LINE_CAP_BUTT")] Butt, + + /// Use a round ending, with the center of the circle at the end point. #[doc(alias = "LINE_CAP_ROUND")] Round, + + /// Use squared-off ending, with the center at the end point. #[doc(alias = "LINE_CAP_SQUARE")] Square, + #[doc(hidden)] __Unknown(i32), } @@ -192,16 +249,29 @@ impl From for LineCap { #[cfg(feature = "use_glib")] gvalue_impl!(LineCap, ffi::gobject::cairo_gobject_line_cap_get_type); +/// Specifies how to render the junction of two lines when stroking. +/// +/// The default line join style is [`LineJoin::Miter`]. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_line_join_t")] pub enum LineJoin { + /// Use a sharp (angled) corner. + /// + /// See [`Context::set_miter_limit`] for more details. + /// + /// [`Context::set_miter_limit`]: crate::Context::set_miter_limit #[doc(alias = "LINE_JOIN_MITER")] Miter, + + /// Use a rounded join, with the center of the circle at the join point. #[doc(alias = "LINE_JOIN_ROUND")] Round, + + /// Use a cut-off join, with the join cut off at half the line width from the join point. #[doc(alias = "LINE_JOIN_BEVEL")] Bevel, + #[doc(hidden)] __Unknown(i32), } @@ -233,72 +303,144 @@ impl From for LineJoin { #[cfg(feature = "use_glib")] gvalue_impl!(LineJoin, ffi::gobject::cairo_gobject_line_join_get_type); +/// Set the compositing operator to be used for all drawing operations. +/// +/// Operators marked as **unbounded** will modify their destination, even outside of the mask +/// layer. Their effect can still be limited by clipping. +/// +/// For a detailed discussion of the effects of each operator, see the [Cairo operator +/// documentation](https://www.cairographics.org/operators/). +/// +/// The default operator is [`Operator::Over`]. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_operator_t")] pub enum Operator { + /// Clear the destination layer. #[doc(alias = "OPERATOR_CLEAR")] Clear, + /// Replace the destination layer. #[doc(alias = "OPERATOR_SOURCE")] Source, + + /// Draw the source layer on top of the destination layer. #[doc(alias = "OPERATOR_OVER")] Over, + + /// **Unbounded**: Draw the source where there was destination content. #[doc(alias = "OPERATOR_IN")] In, + + /// **Unbounded**: Draw the source where there was no destination content. #[doc(alias = "OPERATOR_OUT")] Out, + + /// Draw the source on top of the destination and only there. #[doc(alias = "OPERATOR_ATOP")] Atop, + /// Ignore the source, drawing only destination content. #[doc(alias = "OPERATOR_DEST")] Dest, + + /// Ignore the destination, drawing only source content. #[doc(alias = "OPERATOR_DEST_OVER")] DestOver, + + /// Leave destination content only where there was source content. #[doc(alias = "OPERATOR_DEST_IN")] DestIn, + + /// Leave destination content only where there was no source content. #[doc(alias = "OPERATOR_DEST_OUT")] DestOut, + + /// Leave destination content on top of the source and only there. #[doc(alias = "OPERATOR_DEST_ATOP")] DestAtop, + /// Source and destination are shown only where there is no overlap. #[doc(alias = "OPERATOR_XOR")] Xor, + + /// Source and destination layers are accumulated. #[doc(alias = "OPERATOR_ADD")] Add, + + /// Like [`Operator::Over`], but assumes the source and destination are disjoint geometries. #[doc(alias = "OPERATOR_SATURATE")] Saturate, + /// The source and destination layers are multiplied. This causes the result to be at least as + /// dark as the darker inputs. #[doc(alias = "OPERATOR_MULTIPLY")] Multiply, + + /// The source and destination layers are complemented and multiplied. This causes the result + /// to be at least as light as the lighter inputs. #[doc(alias = "OPERATOR_SCREEN")] Screen, + + /// Chooses [`Operator::Multiply`] or [`Operator::Screen`], depending on the lightness of the + /// destination color. #[doc(alias = "OPERATOR_OVERLAY")] Overlay, + + /// Replaces the destination layer with the source layer if the source layer is darker; + /// otherwise, keeps the source layer. #[doc(alias = "OPERATOR_DARKEN")] Darken, + + /// Replaces the destination layer with the source layer if the source layer is lighter; + /// otherwise, keeps the source layer. #[doc(alias = "OPERATOR_LIGHTEN")] Lighten, + + /// Brightens the destination color to reflect the source color. #[doc(alias = "OPERATOR_COLOR_DODGE")] ColorDodge, + + /// Darkens the destination color to reflect the source color. #[doc(alias = "OPERATOR_COLOR_BURN")] ColorBurn, + + /// Chooses [`Operator::Multiply`] or [`Operator::Screen`], depending on the source color. #[doc(alias = "OPERATOR_HARD_LIGHT")] HardLight, + + /// Chooses [`Operator::ColorDodge`] or [`Operator::ColorBurn`], depending on the source color. #[doc(alias = "OPERATOR_SOFT_LIGHT")] SoftLight, + + /// Takes the difference of the source and destination color. #[doc(alias = "OPERATOR_DIFFERENCE")] Difference, + + /// Like [`Operator::Difference`], but with lower contrast. #[doc(alias = "OPERATOR_EXCLUSION")] Exclusion, + + /// Creates a color with the hue of the source, and the saturation and luminosity of the + /// destination. #[doc(alias = "OPERATOR_HSL_HUE")] HslHue, + + /// Creates a color with the saturation of the source, and the hue and luminosity of the + /// destination. #[doc(alias = "OPERATOR_HSL_SATURATION")] HslSaturation, + + /// Creates a color with the hue and saturation of the source, and the luminosity of the + /// destination. #[doc(alias = "OPERATOR_HSL_COLOR")] HslColor, + + /// Creates a color with the luminosity of the source, and the hue and saturation of the + /// destination. #[doc(alias = "OPERATOR_HSL_LUMINOSITY")] HslLuminosity, + #[doc(hidden)] __Unknown(i32), } @@ -382,6 +524,9 @@ impl From for Operator { #[cfg(feature = "use_glib")] gvalue_impl!(Operator, ffi::gobject::cairo_gobject_operator_get_type); +/// Describes the type of a portion of a [`Path`]. +/// +/// [`Path`]: crate::Path #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_path_data_type_t")] @@ -430,16 +575,31 @@ gvalue_impl!( ffi::gobject::cairo_gobject_path_data_type_get_type ); +/// Describes the content a [`Surface`] will contain. +/// +/// [`Surface`]: crate::Surface #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_content_t")] pub enum Content { + /// The [`Surface`] will hold color content only. + /// + /// [`Surface`]: crate::Surface #[doc(alias = "CONTENT_COLOR")] Color, + + /// The [`Surface`] will hold alpha content only. + /// + /// [`Surface`]: crate::Surface #[doc(alias = "CONTENT_ALPHA")] Alpha, + + /// The [`Surface`] will hold both color and alpha content. + /// + /// [`Surface`]: crate::Surface #[doc(alias = "CONTENT_COLOR_ALPHA")] ColorAlpha, + #[doc(hidden)] __Unknown(i32), } @@ -471,18 +631,33 @@ impl From for Content { #[cfg(feature = "use_glib")] gvalue_impl!(Content, ffi::gobject::cairo_gobject_content_get_type); +/// Describes how pattern color / alpha is determined for areas "outside" the pattern's natural +/// area, (for example, outside the surface bounds or outside the gradient geometry). +/// +/// Mesh patterns are not affected by this setting. +/// +/// The default extend mode is [`Extend::None`] for surface patterns and [`Extend::Pad`] for +/// gradient patterns. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_extend_t")] pub enum Extend { + /// Pixels outside of the source pattern are fully transparent. #[doc(alias = "EXTEND_NONE")] None, + + /// The pattern is tiled by repeating. #[doc(alias = "EXTEND_REPEAT")] Repeat, + + /// The pattern is tiled by reflecting at the edges. #[doc(alias = "EXTEND_REFLECT")] Reflect, + + /// Pixels outside of the pattern copy the closest pixel from the source. #[doc(alias = "EXTEND_PAD")] Pad, + #[doc(hidden)] __Unknown(i32), } @@ -516,22 +691,37 @@ impl From for Extend { #[cfg(feature = "use_glib")] gvalue_impl!(Extend, ffi::gobject::cairo_gobject_extend_get_type); +/// Indicates the filtering to apply when reading pixel values from [`Pattern`]s. +/// +/// [`Pattern`]: crate::Pattern #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_filter_t")] pub enum Filter { + /// High-performance filter with quality similar to [`Filter::Nearest`]. #[doc(alias = "FILTER_FAST")] Fast, + + /// Reasonable-performance filter with quality similar to [`Filter::Bilinear`]. #[doc(alias = "FILTER_GOOD")] Good, + + /// Highest-quality filter; performance may not be suitable for interactive use. #[doc(alias = "FILTER_BEST")] Best, + + /// Nearest-neighbor filtering. #[doc(alias = "FILTER_NEAREST")] Nearest, + + /// Linear interpolation in two dimensions. #[doc(alias = "FILTER_BILINEAR")] Bilinear, + + /// Gaussian interpolation in two dimensions. #[doc(alias = "FILTER_GAUSSIAN")] Gaussian, + #[doc(hidden)] __Unknown(i32), } @@ -569,22 +759,53 @@ impl From for Filter { #[cfg(feature = "use_glib")] gvalue_impl!(Filter, ffi::gobject::cairo_gobject_filter_get_type); +/// Describes the type of a [`Pattern`]. +/// +/// If you are looking to create a specific type of pattern, you should use the appropriate +/// constructor method with one of the `struct`s below: +/// +/// - [`SolidPattern`] +/// - [`SurfacePattern`] +/// - [`LinearGradient`] +/// - [`RadialGradient`] +/// - [`Mesh`] +/// +/// [`Pattern`]: crate::Pattern +/// [`SolidPattern`]: crate::SolidPattern +/// [`SurfacePattern`]: crate::SurfacePattern +/// [`LinearGradient`]: crate::LinearGradient +/// [`RadialGradient`]: crate::RadialGradient +/// [`Mesh`]: crate::Mesh #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_pattern_type_t")] pub enum PatternType { + /// The pattern is a solid opaque / translucent color. #[doc(alias = "PATTERN_TYPE_SOLID")] Solid, + + /// The pattern is based on a [`Surface`]. + /// + /// [`Surface`]: crate::Surface #[doc(alias = "PATTERN_TYPE_SURFACE")] Surface, + + /// The pattern is a linear gradient. #[doc(alias = "PATTERN_TYPE_LINEAR_GRADIENT")] LinearGradient, + + /// The pattern is a radial gradient. #[doc(alias = "PATTERN_TYPE_RADIAL_GRADIENT")] RadialGradient, + + /// The pattern is a mesh. #[doc(alias = "PATTERN_TYPE_MESH")] Mesh, + + /// The pattern is a user providing raster data. #[doc(alias = "PATTERN_TYPE_RASTER_SOURCE")] RasterSource, + #[doc(hidden)] __Unknown(i32), } @@ -625,16 +846,23 @@ gvalue_impl!( ffi::gobject::cairo_gobject_pattern_type_get_type ); +/// Variants of a font face based on their slant. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_font_slant_t")] pub enum FontSlant { + /// Text is rendered upright. #[doc(alias = "FONT_SLANT_NORMAL")] Normal, + + /// Text is rendered with an italic slant. #[doc(alias = "FONT_SLANT_ITALIC")] Italic, + + /// Text is rendered with an oblique slant. #[doc(alias = "FONT_SLANT_OBLIQUE")] Oblique, + #[doc(hidden)] __Unknown(i32), } @@ -666,14 +894,19 @@ impl From for FontSlant { #[cfg(feature = "use_glib")] gvalue_impl!(FontSlant, ffi::gobject::cairo_gobject_font_slant_get_type); +/// Variants of a font face based on their weight. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_font_weight_t")] pub enum FontWeight { + /// Normal font weight. #[doc(alias = "FONT_WEIGHT_NORMAL")] Normal, + + /// Bolded font weight. #[doc(alias = "FONT_WEIGHT_BOLD")] Bold, + #[doc(hidden)] __Unknown(i32), } @@ -703,14 +936,19 @@ impl From for FontWeight { #[cfg(feature = "use_glib")] gvalue_impl!(FontWeight, ffi::gobject::cairo_gobject_font_weight_get_type); +/// Specifies properties of a text cluster mapping. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_text_cluster_flags_t")] pub enum TextClusterFlags { + /// The clusters in the cluster array map to glyphs in the glyph array from start to end. #[doc(alias = "TEXT_CLUSTER_FLAGS_NONE")] None, + + /// The clusters in the cluster array map to glyphs in the glyph array from end to start. #[doc(alias = "TEXT_CLUSTER_FLAGS_BACKWARD")] Backward, + #[doc(hidden)] __Unknown(i32), } @@ -743,24 +981,45 @@ gvalue_impl!( ffi::gobject::cairo_gobject_text_cluster_flags_get_type ); +/// Describes the type of a [`FontFace`] or [`ScaledFont`], also known as "font backends" within +/// Cairo. +/// +/// [`FontFace`]: crate::FontFace +/// [`ScaledFont`]: crate::ScaledFont #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_font_type_t")] pub enum FontType { + /// The font was created using [`FontFace::toy_create`]. + /// + /// [`FontFace::toy_create`]: crate::FontFace::toy_create #[doc(alias = "FONT_TYPE_FONT_TYPE_TOY")] FontTypeToy, + + /// The font is of type FreeType. #[doc(alias = "FONT_TYPE_FONT_TYPE_FT")] FontTypeFt, + + /// The font is of type Win32. #[doc(alias = "FONT_TYPE_FONT_TYPE_WIN32")] FontTypeWin32, + + /// The font is of type Quartz. #[doc(alias = "FONT_TYPE_FONT_TYPE_QUARTZ")] FontTypeQuartz, + + /// The font is a [`UserFontFace`]. + /// + /// [`UserFontFace`]: crate::UserFontFace #[doc(alias = "FONT_TYPE_FONT_TYPE_USER")] FontTypeUser, + + /// The font is of type Win32 DWrite. #[cfg(feature = "v1_18")] #[cfg_attr(docsrs, doc(cfg(feature = "v1_18")))] #[doc(alias = "FONT_TYPE_FONT_TYPE_DWRITE")] FontTypeDwrite, + #[doc(hidden)] __Unknown(i32), } @@ -800,20 +1059,32 @@ impl From for FontType { #[cfg(feature = "use_glib")] gvalue_impl!(FontType, ffi::gobject::cairo_gobject_font_type_get_type); +/// Specifies the order of color elements within each pixel on the display +/// device, when rendering with [`Antialias::Subpixel`]. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_subpixel_order_t")] pub enum SubpixelOrder { + /// Use the default subpixel order for for the target device. #[doc(alias = "SUBPIXEL_ORDER_DEFAULT")] Default, + + /// Subpixel elements are arranged horizontally with red at the left. #[doc(alias = "SUBPIXEL_ORDER_RGB")] Rgb, + + /// Subpixel elements are arranged horizontally with blue at the left. #[doc(alias = "SUBPIXEL_ORDER_BGR")] Bgr, + + /// Subpixel elements are arranged vertically with red at the top. #[doc(alias = "SUBPIXEL_ORDER_VRGB")] Vrgb, + + /// Subpixel elements are arranged vertically with blue at the top. #[doc(alias = "SUBPIXEL_ORDER_VBGR")] Vbgr, + #[doc(hidden)] __Unknown(i32), } @@ -852,20 +1123,38 @@ gvalue_impl!( ffi::gobject::cairo_gobject_subpixel_order_get_type ); +/// Specify the type of hinting to do on font outlines. +/// +/// Hinting is the process of fitting outlines to the pixel grid in order to improve the appearance +/// of the result. Since hinting outlines involves distorting them, it also reduces the +/// faithfulness to the original outline shapes. +/// +/// Not all of the outline hinting styles are supported by all font backends. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_hint_style_t")] pub enum HintStyle { + /// Use the default hint style for the font backend and target device. #[doc(alias = "HINT_STYLE_DEFAULT")] Default, + + /// Do not hint outlines. #[doc(alias = "HINT_STYLE_NONE")] None, + + /// Hint outlines slightly to improve contrast while retaining good fidelity to the original. #[doc(alias = "HINT_STYLE_SLIGHT")] Slight, + + /// Hint outlines with medium strength giving a compromise between fidelity to the original and + /// contrast. #[doc(alias = "HINT_STYLE_MEDIUM")] Medium, + + /// Hint outlines to maximize contrast. #[doc(alias = "HINT_STYLE_FULL")] Full, + #[doc(hidden)] __Unknown(i32), } @@ -901,16 +1190,27 @@ impl From for HintStyle { #[cfg(feature = "use_glib")] gvalue_impl!(HintStyle, ffi::gobject::cairo_gobject_hint_style_get_type); +/// Whether to hint font metrics. +/// +/// Hinting font metrics means quantizing them so that they are integer values in device space. +/// Doing this improves the consistency of letter and line spacing, however it also means that text +/// will be laid out differently at different zoom factors. #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_hint_metrics_t")] pub enum HintMetrics { + /// Hint metrics in the default manner for the font backend and target device. #[doc(alias = "HINT_METRICS_DEFAULT")] Default, + + /// Do not hint font metrics. #[doc(alias = "HINT_METRICS_OFF")] Off, + + /// Do hint font metrics. #[doc(alias = "HINT_METRICS_ON")] On, + #[doc(hidden)] __Unknown(i32), } @@ -945,58 +1245,146 @@ gvalue_impl!( ffi::gobject::cairo_gobject_hint_metrics_get_type ); + +/// Describes the type of a [`Surface`], also known as "surface backends" within Cairo. +/// +/// If you are looking to create a specific type of surface, you should use the appropriate +/// constructor method with one of the `struct`s below: +/// +/// - [`ImageSurface`] +/// - [`PdfSurface`] +/// - [`PsSurface`] +/// - [`SvgSurface`] +/// - [`XCBSurface`] +/// +/// Note that each surface, with the exception of [`ImageSurface`], has a corresponding feature +/// flag that must be enabled in order to use it. +/// +/// [`Surface`]: crate::Surface +/// [`ImageSurface`]: crate::ImageSurface +/// [`PdfSurface`]: crate::PdfSurface +/// [`PsSurface`]: crate::PsSurface +/// [`SvgSurface`]: crate::SvgSurface +/// [`XCBSurface`]: crate::XCBSurface #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_surface_type_t")] pub enum SurfaceType { + /// Type image. #[doc(alias = "SURFACE_TYPE_IMAGE")] Image, + + /// Type `pdf`. #[doc(alias = "SURFACE_TYPE_PDF")] Pdf, + + /// Type `ps`. #[doc(alias = "SURFACE_TYPE_PS")] Ps, + + /// Type `xlib`. #[doc(alias = "SURFACE_TYPE_XLIB")] Xlib, + + /// Type `xcb`. #[doc(alias = "SURFACE_TYPE_XCB")] Xcb, + + /// Type `glitz`. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_GLITZ")] Glitz, + + /// Type `quartz`. #[doc(alias = "SURFACE_TYPE_QUARTZ")] Quartz, + + /// Type `win32`. #[doc(alias = "SURFACE_TYPE_WIN32")] Win32, + + /// Type `beos`. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_BE_OS")] BeOs, + + /// Type `directfb`. #[doc(alias = "SURFACE_TYPE_DIRECT_FB")] DirectFb, + + /// Type `svg`. #[doc(alias = "SURFACE_TYPE_SVG")] Svg, + + /// Type `os2`. #[doc(alias = "SURFACE_TYPE_OS2")] Os2, + + /// Win32 printing surface. #[doc(alias = "SURFACE_TYPE_WIN32_PRINTING")] Win32Printing, + + /// Type `quartz_image`. #[doc(alias = "SURFACE_TYPE_QUARTZ_IMAGE")] QuartzImage, + + /// Type `script`. #[doc(alias = "SURFACE_TYPE_SCRIPT")] Script, + + /// Type `qt`. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_QT")] Qt, + + /// Type `recording`. #[doc(alias = "SURFACE_TYPE_RECORDING")] Recording, + + /// OpenVG drawing surface. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_VG")] Vg, + + /// OpenGL surface. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_GL")] Gl, + + /// Direct Render Manager surface. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_DRM")] Drm, + + /// Type `tee`. #[doc(alias = "SURFACE_TYPE_TEE")] Tee, + + /// Type `xml`. #[doc(alias = "SURFACE_TYPE_XML")] Xml, + + /// Type `skia`. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_SKIA")] Skia, + + /// Subsurface created with [`Surface::create_for_rectangle`]. + /// + /// [`Surface::create_for_rectangle`]: crate::Surface::create_for_rectangle #[doc(alias = "SURFACE_TYPE_SUBSURFACE")] Subsurface, + + /// Type `cogl`. + /// + /// This surface type is deprecated and will never be set by Cairo. #[doc(alias = "SURFACE_TYPE_COGL")] Cogl, #[doc(hidden)] @@ -1077,32 +1465,59 @@ gvalue_impl!( ffi::gobject::cairo_gobject_surface_type_get_type ); +/// Units of measurement that can be used for various coordinates and lengths in the SVG +/// specification, and when drawing to an [`SvgSurface`]. +/// +/// [`SvgSurface`]: crate::SvgSurface #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[cfg(all(feature = "svg", feature = "v1_16"))] #[cfg_attr(docsrs, doc(cfg(all(feature = "svg", feature = "v1_16"))))] #[non_exhaustive] #[doc(alias = "cairo_svg_unit_t")] pub enum SvgUnit { + /// User-specified unit; a value in the current coordinate system. + /// + /// If used in the root element for the initial coordinate system, it corresponds to + /// [`SvgUnit::Px`]. #[doc(alias = "SVG_UNIT_USER")] User, + + /// The size of the element's font. #[doc(alias = "SVG_UNIT_EM")] Em, + + /// The x-height of the element's font. #[doc(alias = "SVG_UNIT_EX")] Ex, + + /// Pixels. `1px` is equal to `1/96th` of an inch. #[doc(alias = "SVG_UNIT_PX")] Px, + + /// Inches. `1in` is equal to `2.54cm`. #[doc(alias = "SVG_UNIT_IN")] In, + + /// Centimeters. `1cm` is equal to `96px/2.54`. #[doc(alias = "SVG_UNIT_CM")] Cm, + + /// Millimeters. `1mm` is equal to `1/10th of 1cm`. #[doc(alias = "SVG_UNIT_MM")] Mm, + + /// Points. `1pt` is equal to `1/72nd of 1in`. #[doc(alias = "SVG_UNIT_PT")] Pt, + + /// Picas. `1pc` is equal to `12pt`. #[doc(alias = "SVG_UNIT_PC")] Pc, + + /// Some fraction of another reference value. #[doc(alias = "SVG_UNIT_PERCENT")] Percent, + #[doc(hidden)] __Unknown(i32), } @@ -1149,24 +1564,49 @@ impl From for SvgUnit { } } +/// Identifies the memory format of image data. #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_format_t")] pub enum Format { + /// No such format exists or is supported. #[doc(alias = "FORMAT_INVALID")] Invalid, + + /// Each pixel is a 32-bit quantity stored native-endian, with alpha in the upper 8 bits, then + /// red, then green, then blue. + /// + /// Pre-multiplied alpha is used (i.e. 50% transparent red is `0x80800000`, not `0x80ff0000`). #[doc(alias = "FORMAT_A_RGB32")] ARgb32, + + /// Each pixel is a 32-bit quantity stored native-endian, with the upper 8 bits unused, and the + /// remaining 24 bits containing red, green, and blue, in that order. #[doc(alias = "FORMAT_RGB24")] Rgb24, + + /// Each pixel is a 8-bit quantity holding an alpha value. #[doc(alias = "FORMAT_A8")] A8, + + /// Each pixel is a 1-bit quantity holding an alpha value. + /// + /// Pixels are packed together into 32-bit quantities. The ordering of the bits matches the + /// endianness of the platform. On a big-endian machine, the first pixel is in the uppermost + /// bit, and on a little-endian machine, the first pixel is in the least-significant bit. #[doc(alias = "FORMAT_A1")] A1, + + /// Each pixel is a 16-bit quantity with red in the upper 5 bits, green in the middle 6 + /// bits, and blue in the lower 5 bits. #[doc(alias = "FORMAT_RGB16_565")] Rgb16_565, + + /// Each pixel is a 30-bit quantity stored native-endian, with the 30 bits split evenly between + /// red, green, and blue, in that order. #[doc(alias = "FORMAT_RGB30")] Rgb30, + #[doc(hidden)] __Unknown(i32), } @@ -1207,6 +1647,8 @@ impl From for Format { gvalue_impl!(Format, ffi::gobject::cairo_gobject_format_get_type); impl Format { + /// Provides a stride value that will respect all alignment requirements of the accelerated + /// image-rendering code within Cairo. #[doc(alias = "cairo_format_stride_for_width")] pub fn stride_for_width(self, width: u32) -> Result { assert!(width <= i32::MAX as u32); @@ -1221,16 +1663,25 @@ impl Format { } } +/// The result of [`Region::contains_rectangle`]. +/// +/// [`Region::contains_rectangle`]: crate::Region::contains_rectangle #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_region_overlap_t")] pub enum RegionOverlap { + /// The contents are entirely inside the region. #[doc(alias = "REGION_OVERLAP_IN")] In, + + /// The contents are entirely outside the region. #[doc(alias = "REGION_OVERLAP_OUT")] Out, + + /// Some contents are inside and some are outside the region. #[doc(alias = "REGION_OVERLAP_PART")] Part, + #[doc(hidden)] __Unknown(i32), } @@ -1266,37 +1717,58 @@ gvalue_impl!( ); bitflags::bitflags! { + /// Attributes of an outline item. #[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)] pub struct PdfOutline: i32 { + /// The outline item defaults to being open in the PDF viewer. #[doc(alias = "PDF_OUTLINE_FLAG_OPEN")] const OPEN = ffi::PDF_OUTLINE_FLAG_OPEN; + + /// The outline item is displayed in bold text. #[doc(alias = "PDF_OUTLINE_FLAG_BOLD")] const BOLD = ffi::PDF_OUTLINE_FLAG_BOLD; + + /// The outline item is displayed in italic text. #[doc(alias = "PDF_OUTLINE_FLAG_ITALIC")] const ITALIC = ffi::PDF_OUTLINE_FLAG_ITALIC; } } +/// Specify which part of a PDF document's metadata to set. #[cfg(feature = "pdf")] #[cfg_attr(docsrs, doc(cfg(feature = "pdf")))] #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_pdf_metadata_t")] pub enum PdfMetadata { + /// The document's title. #[doc(alias = "PDF_METADATA_TITLE")] Title, + + /// The document's author. #[doc(alias = "PDF_METADATA_AUTHOR")] Author, + + /// The document's subject. #[doc(alias = "PDF_METADATA_SUBJECT")] Subject, + + /// The document's keywords. #[doc(alias = "PDF_METADATA_KEYWORDS")] Keywords, + + /// The document's creator. #[doc(alias = "PDF_METADATA_CREATOR")] Creator, + + /// The document's creation date. #[doc(alias = "PDF_METADATA_CREATE_DATE")] CreateDate, + + /// The document's modification date. #[doc(alias = "PDF_METADATA_MOD_DATE")] ModDate, + #[doc(hidden)] __Unknown(i32), } @@ -1337,24 +1809,33 @@ impl From for PdfMetadata { } } +/// Specify the version of the PDF specification to use when generating a PDF document. #[cfg(feature = "pdf")] #[cfg_attr(docsrs, doc(cfg(feature = "pdf")))] #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_pdf_version_t")] pub enum PdfVersion { + /// Version 1.4 of the PDF specification. #[doc(alias = "PDF_VERSION__1_4")] _1_4, + + /// Version 1.5 of the PDF specification. #[doc(alias = "PDF_VERSION__1_5")] _1_5, + + /// Version 1.6 of the PDF specification. #[cfg(feature = "v1_18")] #[cfg_attr(docsrs, doc(cfg(feature = "v1_18")))] #[doc(alias = "PDF_VERSION__1_6")] _1_6, + + /// Version 1.7 of the PDF specification. #[cfg(feature = "v1_18")] #[cfg_attr(docsrs, doc(cfg(feature = "v1_18")))] #[doc(alias = "PDF_VERSION__1_7")] _1_7, + #[doc(hidden)] __Unknown(i32), } @@ -1393,16 +1874,21 @@ impl From for PdfVersion { } } +/// Specify the version of the SVG specification to use when generating an SVG document. #[cfg(feature = "svg")] #[cfg_attr(docsrs, doc(cfg(feature = "svg")))] #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_svg_version_t")] pub enum SvgVersion { + /// Version 1.1 of the SVG specification. #[doc(alias = "SVG_VERSION__1_1")] _1_1, + + /// Version 1.2 of the SVG specification. #[doc(alias = "SVG_VERSION__1_2")] _1_2, + #[doc(hidden)] __Unknown(i32), } @@ -1433,16 +1919,22 @@ impl From for SvgVersion { } } +/// Specify the language level of the PostScript Language Reference to use when generating a +/// PostScript document. #[cfg(feature = "ps")] #[cfg_attr(docsrs, doc(cfg(feature = "ps")))] #[derive(Clone, Copy, Debug, PartialEq, Eq)] #[non_exhaustive] #[doc(alias = "cairo_ps_level_t")] pub enum PsLevel { + /// Language level 2 of the PostScript specification. #[doc(alias = "PS_LEVEL__2")] _2, + + /// Language level 3 of the PostScript specification. #[doc(alias = "PS_LEVEL__3")] _3, + #[doc(hidden)] __Unknown(i32), } @@ -1473,18 +1965,29 @@ impl From for PsLevel { } } +/// Specify which control point of a [`Mesh`] to set or get. +/// +/// [`Mesh`]: crate::Mesh #[derive(Clone, PartialEq, Eq, PartialOrd, Copy, Debug)] #[non_exhaustive] #[doc(alias = "cairo_mesh_corner_t")] pub enum MeshCorner { + /// Mesh corner 0 (the first control point). #[doc(alias = "MESH_CORNER_MESH_CORNER0")] MeshCorner0, + + /// Mesh corner 1. #[doc(alias = "MESH_CORNER_MESH_CORNER1")] MeshCorner1, + + /// Mesh corner 2. #[doc(alias = "MESH_CORNER_MESH_CORNER2")] MeshCorner2, + + /// Mesh corner 3 (the last control point). #[doc(alias = "MESH_CORNER_MESH_CORNER3")] MeshCorner3, + #[doc(hidden)] __Unknown(u32), } @@ -1515,16 +2018,29 @@ impl From for MeshCorner { } } +/// Flags to control how FreeType renders the glyphs for a particular [`FontFace`]. +/// +/// FreeType provides the ability to synthesize different glyphs from a base font, which is useful +/// if you lack those glyphs from a true bold or oblique font. +/// +/// Note that when synthesizing glyphs, any generated [`FontExtents`] will only be estimates. +/// +/// [`FontFace`]: crate::FontFace +/// [`FontExtents`]: crate::FontExtents #[cfg(feature = "freetype")] #[cfg_attr(docsrs, doc(cfg(feature = "freetype")))] #[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_ft_synthesize_t")] pub enum FtSynthesize { + /// Embolden the glyphs (redraw them with a pixel offset). #[doc(alias = "CAIRO_FT_SYNTHESIZE_BOLD")] Bold, + + /// Slant the glyphs 12 degrees to the right. #[doc(alias = "CAIRO_FT_SYNTHESIZE_OBLIQUE")] Oblique, + #[doc(hidden)] __Unknown(u32), } @@ -1555,16 +2071,21 @@ impl From for FtSynthesize { } } +/// Possible output variants when drawing to a script surface. #[cfg(feature = "script")] #[cfg_attr(docsrs, doc(cfg(feature = "script")))] #[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_script_mode_t")] pub enum ScriptMode { + /// The output will be in readable text. #[doc(alias = "CAIRO_SCRIPT_MODE_ASCII")] Ascii, + + /// The output will use byte codes. #[doc(alias = "CAIRO_SCRIPT_MODE_BINARY")] Binary, + #[doc(hidden)] __Unknown(i32), } @@ -1595,28 +2116,49 @@ impl From for ScriptMode { } } +/// Specifies the type of a given [`Device`], also known as "backends" within Cairo. +/// +/// [`Device`]: crate::Device #[derive(Clone, PartialEq, Eq, PartialOrd, Debug, Copy)] #[non_exhaustive] #[doc(alias = "cairo_device_type_t")] pub enum DeviceType { + /// Type Direct Render Manager. #[doc(alias = "CAIRO_DEVICE_TYPE_DRM")] - Ascii, + Drm, + + /// Type OpenGL. #[doc(alias = "CAIRO_DEVICE_TYPE_GL")] - Binary, + Gl, + + /// Type script. #[doc(alias = "CAIRO_DEVICE_TYPE_SCRIPT")] Script, + + /// Type XCB. #[doc(alias = "CAIRO_DEVICE_TYPE_XCB")] Xcb, + + /// Type Xlib. #[doc(alias = "CAIRO_DEVICE_TYPE_XLIB")] Xlib, + + /// Type XML. #[doc(alias = "CAIRO_DEVICE_TYPE_XML")] Xml, + + /// Type Cogl. #[doc(alias = "CAIRO_DEVICE_TYPE_COGL")] Cogl, + + /// Type Win32. #[doc(alias = "CAIRO_DEVICE_TYPE_WIN32")] Win32, + + /// Invalid type. #[doc(alias = "CAIRO_DEVICE_TYPE_INVALID")] Invalid, + #[doc(hidden)] __Unknown(i32), } @@ -1625,8 +2167,8 @@ pub enum DeviceType { impl From for ffi::cairo_device_type_t { fn from(val: DeviceType) -> ffi::cairo_device_type_t { match val { - DeviceType::Ascii => ffi::CAIRO_DEVICE_TYPE_DRM, - DeviceType::Binary => ffi::CAIRO_DEVICE_TYPE_GL, + DeviceType::Drm => ffi::CAIRO_DEVICE_TYPE_DRM, + DeviceType::Gl => ffi::CAIRO_DEVICE_TYPE_GL, DeviceType::Script => ffi::CAIRO_DEVICE_TYPE_SCRIPT, DeviceType::Xcb => ffi::CAIRO_DEVICE_TYPE_XCB, DeviceType::Xlib => ffi::CAIRO_DEVICE_TYPE_XLIB, @@ -1643,8 +2185,8 @@ impl From for ffi::cairo_device_type_t { impl From for DeviceType { fn from(value: ffi::cairo_device_type_t) -> Self { match value { - ffi::CAIRO_DEVICE_TYPE_DRM => Self::Ascii, - ffi::CAIRO_DEVICE_TYPE_GL => Self::Binary, + ffi::CAIRO_DEVICE_TYPE_DRM => Self::Drm, + ffi::CAIRO_DEVICE_TYPE_GL => Self::Gl, ffi::CAIRO_DEVICE_TYPE_SCRIPT => Self::Script, ffi::CAIRO_DEVICE_TYPE_XCB => Self::Xcb, ffi::CAIRO_DEVICE_TYPE_XLIB => Self::Xlib, diff --git a/cairo/src/paths.rs b/cairo/src/paths.rs index f870d3165905..e6106ea338dc 100644 --- a/cairo/src/paths.rs +++ b/cairo/src/paths.rs @@ -4,6 +4,63 @@ use std::{iter::FusedIterator, ptr}; use crate::{ffi, PathDataType}; +/// A path of segments, representing a series of moveto, lineto, curveto, and closepath operations +/// that define a path to draw. +/// +/// A [`Path`] can be thought of as a list of commands that define how to draw a particular shape, +/// akin to drawing a shape with pen and paper. Each command is represented by a [`PathSegment`], +/// which define a specific operation to perform with the pen. +/// +/// Paths are created by calling [`Context::copy_path`] and [`Context::copy_path_flat`], which +/// returns a copy of the current path that the context is using. This allows you to inspect the +/// path that has been drawn so far. +/// +/// # Examples +/// +/// ``` +/// # fn main() -> Result<(), cairo::Error> { +/// # use cairo::{Context, Format, ImageSurface}; +/// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; +/// # let ctx = Context::new(&surface)?; +/// // paint the background black +/// ctx.paint()?; +/// +/// // draw a 5-pointed star with a white fill, in similar manner to how a human would +/// let angle = std::f64::consts::TAU / 2.5; +/// ctx.set_source_rgb(1.0, 1.0, 1.0); +/// for i in 0..5 { +/// let x = 50.0 + 40.0 * (i as f64 * angle).cos(); +/// let y = 50.0 + 40.0 * (i as f64 * angle).sin(); +/// if i == 0 { +/// ctx.move_to(x, y); +/// } else { +/// ctx.line_to(x, y); +/// } +/// } +/// ctx.close_path(); +/// +/// // print the path details before filling (will clear the context path) +/// let path = ctx.copy_path()?; +/// for element in path.iter() { +/// println!("{:?}", element); +/// } +/// +/// ctx.fill()?; +/// +/// // output: +/// // MoveTo((90.0, 50.0)) +/// // LineTo((17.640625, 73.51171875)) +/// // LineTo((62.359375, 11.95703125)) +/// // LineTo((62.359375, 88.04296875)) +/// // LineTo((17.640625, 26.48828125)) +/// // ClosePath +/// // MoveTo((90.0, 50.0)) +/// # Ok(()) +/// # } +/// ``` +/// +/// [`Context::copy_path`]: crate::Context::copy_path +/// [`Context::copy_path_flat`]: crate::Context::copy_path_flat #[derive(Debug)] #[doc(alias = "cairo_path_t")] pub struct Path(ptr::NonNull); @@ -20,6 +77,7 @@ impl Path { Path(ptr::NonNull::new_unchecked(pointer)) } + /// Returns an iterator over the segments of the path. pub fn iter(&self) -> PathSegments { use std::slice; @@ -51,14 +109,31 @@ impl Drop for Path { } } +/// A segment of a path, representing a single moveto, lineto, curveto, or closepath operation. +/// +/// See the documentation for [`Path`] for more information. #[derive(Debug, Clone, Copy, PartialEq)] pub enum PathSegment { + /// Lift up the "pen" and move it to the given point, starting a new subpath. MoveTo((f64, f64)), + + /// Draw a straight line from the current point to the given point. LineTo((f64, f64)), + + /// Draw a cubic Bézier curve from the current point to the given point, using the two control + /// points to define the curve. + /// + /// The first and second points are the control points, and the third point is the end point. CurveTo((f64, f64), (f64, f64), (f64, f64)), + + /// Draw a straight line from the current point to the starting point of the subpath, closing + /// it. ClosePath, } +/// An iterator over the segments of a [`Path`]. +/// +/// This struct is created by the [`Path::iter`] method. pub struct PathSegments<'a> { data: &'a [ffi::cairo_path_data], i: usize,