From 5b155d4e8d7d7b2cef81789cb84a12aa4dfc127a Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Sun, 24 Sep 2023 17:51:25 -0500 Subject: [PATCH 1/7] Document Context, source and stack functions --- cairo/src/context.rs | 218 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 218 insertions(+) diff --git a/cairo/src/context.rs b/cairo/src/context.rs index fd3b7de4ebb8..cfef44cf4bd0 100644 --- a/cairo/src/context.rs +++ b/cairo/src/context.rs @@ -59,6 +59,27 @@ impl fmt::Display 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}; +/// +/// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); +/// let ctx = Context::new(&surface).unwrap(); +/// +/// // paint the background black +/// ctx.set_source_rgb(0.0, 0.0, 0.0); +/// ctx.paint(); +/// +/// // etc. +/// ``` +/// +/// [`ImageSurface`]: crate::ImageSurface +/// [`PdfSurface`]: crate::PdfSurface #[derive(Debug)] #[repr(transparent)] pub struct Context(ptr::NonNull); @@ -163,6 +184,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> { @@ -170,67 +194,260 @@ 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).unwrap(); + /// let ctx = Context::new(&surface).unwrap(); + /// + /// // 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).unwrap(); + /// let ctx = Context::new(&surface).unwrap(); + /// + /// // 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).unwrap(); + /// let ctx = Context::new(&surface).unwrap(); + /// + /// // 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, RadialGradient}; + /// + /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); + /// let ctx = Context::new(&surface).unwrap(); + /// + /// // creates 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); + /// + /// // paints 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(); @@ -509,6 +726,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()) }; From 5a4d01740b5e359bec2de118db1350bbe26aacb3 Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Sun, 24 Sep 2023 18:45:03 -0500 Subject: [PATCH 2/7] Document context, path, pattern enums --- cairo/src/context.rs | 2 + cairo/src/enums.rs | 221 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 223 insertions(+) diff --git a/cairo/src/context.rs b/cairo/src/context.rs index cfef44cf4bd0..3f66f95e1da5 100644 --- a/cairo/src/context.rs +++ b/cairo/src/context.rs @@ -448,6 +448,7 @@ impl Context { /// ctx.paint()?; /// # Ok(()) /// # } + /// ``` #[doc(alias = "cairo_set_source")] pub fn set_source(&self, source: impl AsRef) -> Result<(), Error> { let source = source.as_ref(); @@ -458,6 +459,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 { diff --git a/cairo/src/enums.rs b/cairo/src/enums.rs index 65e4d3e12bd0..cdf55f747abf 100644 --- a/cairo/src/enums.rs +++ b/cairo/src/enums.rs @@ -53,28 +53,59 @@ 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), } @@ -133,14 +164,32 @@ impl fmt::Display 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), } @@ -184,16 +233,25 @@ impl fmt::Display 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), } @@ -240,16 +298,29 @@ impl fmt::Display 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), } @@ -296,72 +367,144 @@ impl fmt::Display 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), } @@ -486,6 +629,9 @@ impl fmt::Display 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")] @@ -550,16 +696,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), } @@ -606,18 +767,33 @@ impl fmt::Display 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), } @@ -667,22 +843,37 @@ impl fmt::Display 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), } @@ -738,20 +929,50 @@ impl fmt::Display 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)] From d91c6d88e5b550db91155fd02586065337ad6fbb Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Tue, 17 Oct 2023 23:47:08 -0500 Subject: [PATCH 3/7] Complete documentation for enums --- cairo/src/enums.rs | 323 ++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 321 insertions(+), 2 deletions(-) diff --git a/cairo/src/enums.rs b/cairo/src/enums.rs index cdf55f747abf..178e452cc55c 100644 --- a/cairo/src/enums.rs +++ b/cairo/src/enums.rs @@ -1033,14 +1033,20 @@ 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)] @@ -1089,12 +1095,16 @@ impl fmt::Display 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)] @@ -1140,12 +1150,16 @@ impl fmt::Display 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)] @@ -1194,20 +1208,40 @@ 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")] @@ -1270,18 +1304,29 @@ impl fmt::Display 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)] @@ -1339,18 +1384,35 @@ 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)] @@ -1405,14 +1467,24 @@ impl fmt::Display 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)] @@ -1464,58 +1536,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)] @@ -1633,30 +1793,56 @@ 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)] @@ -1729,22 +1915,64 @@ impl fmt::Display 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, + /* +CAIRO_FORMAT_ARGB32 +each pixel is a 32-bit quantity, with alpha in the upper 8 bits, then red, then green, then blue. The 32-bit quantities are stored native-endian. Pre-multiplied alpha is used. (That is, 50% transparent red is 0x80800000, not 0x80ff0000.) (Since 1.0) +CAIRO_FORMAT_RGB24 +each pixel is a 32-bit quantity, with the upper 8 bits unused. Red, Green, and Blue are stored in the remaining 24 bits in that order. (Since 1.0) +CAIRO_FORMAT_A8 +each pixel is a 8-bit quantity holding an alpha value. (Since 1.0) +CAIRO_FORMAT_A1 +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, on a little-endian machine the first pixel is in the least-significant bit. (Since 1.0) +CAIRO_FORMAT_RGB16_565 +each pixel is a 16-bit quantity with red in the upper 5 bits, then green in the middle 6 bits, and blue in the lower 5 bits. (Since 1.2) +CAIRO_FORMAT_RGB30 +like RGB24 but with 10bpc. (Since 1.12) +CAIRO_FORMAT_RGB96F +3 floats, R, G, B. (Since 1.17.2) +CAIRO_FORMAT_RGBA128F +4 floats, R, G, B, A. (Since 1.17.2) + */ + + /// 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)] @@ -1806,6 +2034,8 @@ impl fmt::Display 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); @@ -1820,14 +2050,22 @@ 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)] @@ -1880,35 +2118,55 @@ 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)] @@ -1972,20 +2230,28 @@ impl fmt::Display 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")] @@ -2048,14 +2314,18 @@ impl fmt::Display 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)] @@ -2104,14 +2374,19 @@ impl fmt::Display 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)] @@ -2160,16 +2435,26 @@ impl fmt::Display 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)] @@ -2218,14 +2503,26 @@ impl fmt::Display 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)] @@ -2274,14 +2571,18 @@ impl fmt::Display 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)] @@ -2330,26 +2631,44 @@ impl fmt::Display for ScriptMode { } } +/// Specifies the type of a given [`Device`], also known as "backends" within Cairo. #[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)] From 789ad27e4ef4c9e26e1a6e5316ca7550dc359a0a Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Wed, 18 Oct 2023 09:41:08 -0500 Subject: [PATCH 4/7] Missing docitem in DeviceType enum --- cairo/src/enums.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/cairo/src/enums.rs b/cairo/src/enums.rs index 178e452cc55c..89ab26d8ec75 100644 --- a/cairo/src/enums.rs +++ b/cairo/src/enums.rs @@ -2632,6 +2632,8 @@ impl fmt::Display 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")] From a3a0040ebef5e4abc1b33ee396106fbc663ed8b7 Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Sun, 15 Dec 2024 00:40:56 -0600 Subject: [PATCH 5/7] Document path, and Context::close_to due to use in example --- cairo/src/context.rs | 64 ++++++++++++++++++++++++++++++++++++ cairo/src/paths.rs | 77 ++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 141 insertions(+) diff --git a/cairo/src/context.rs b/cairo/src/context.rs index 4a465515384c..0dc22a3874ae 100644 --- a/cairo/src/context.rs +++ b/cairo/src/context.rs @@ -1064,6 +1064,70 @@ 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. + /// + /// **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).unwrap(); + /// let ctx = Context::new(&surface).unwrap(); + /// + /// // 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(()) + /// # } + /// ``` + /// + /// [`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()) } diff --git a/cairo/src/paths.rs b/cairo/src/paths.rs index f870d3165905..b954f1bd3ef9 100644 --- a/cairo/src/paths.rs +++ b/cairo/src/paths.rs @@ -4,6 +4,65 @@ 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).unwrap(); +/// let ctx = Context::new(&surface).unwrap(); +/// +/// // 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 +79,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 +111,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 Bezier 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, From 6fe11829efb6bfdcf7f8b9836def8733b4435599 Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Sun, 15 Dec 2024 01:29:03 -0600 Subject: [PATCH 6/7] Add documentation for Path functions, example for curve_to --- cairo/src/context.rs | 91 ++++++++++++++++++++++++++++++++++++++++++++ cairo/src/paths.rs | 2 +- 2 files changed, 92 insertions(+), 1 deletion(-) diff --git a/cairo/src/context.rs b/cairo/src/context.rs index 0dc22a3874ae..bf4092ea79bb 100644 --- a/cairo/src/context.rs +++ b/cairo/src/context.rs @@ -1076,6 +1076,8 @@ impl Context { /// 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. @@ -1126,6 +1128,7 @@ impl Context { /// # } /// ``` /// + /// [`PathSegment::ClosePath`]: crate::PathSegment::ClosePath /// [`MoveTo`]: crate::PathSegment::MoveTo /// [`ClosePath`]: crate::PathSegment::ClosePath #[doc(alias = "cairo_close_path")] @@ -1143,16 +1146,104 @@ 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).unwrap(); + /// let ctx = Context::new(&surface).unwrap(); + /// + /// // 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/paths.rs b/cairo/src/paths.rs index b954f1bd3ef9..caa7ae47ad2a 100644 --- a/cairo/src/paths.rs +++ b/cairo/src/paths.rs @@ -122,7 +122,7 @@ pub enum PathSegment { /// Draw a straight line from the current point to the given point. LineTo((f64, f64)), - /// Draw a cubic Bezier curve from the current point to the given point, using the two control + /// 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. From 743277d46a31b0fb60652dbc87ece057ec992918 Mon Sep 17 00:00:00 2001 From: Tam Pham Date: Sun, 15 Dec 2024 01:39:11 -0600 Subject: [PATCH 7/7] Remove duplicated setup code from Context examples --- cairo/src/context.rs | 83 ++++++++++++++++++++++++-------------------- cairo/src/paths.rs | 8 ++--- 2 files changed, 48 insertions(+), 43 deletions(-) diff --git a/cairo/src/context.rs b/cairo/src/context.rs index bf4092ea79bb..4869ec608c68 100644 --- a/cairo/src/context.rs +++ b/cairo/src/context.rs @@ -60,14 +60,33 @@ impl fmt::Debug for RectangleList { /// ``` /// use cairo::{Context, Format, ImageSurface}; /// -/// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); -/// let ctx = Context::new(&surface).unwrap(); +/// 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(); +/// // paint the background black +/// ctx.set_source_rgb(0.0, 0.0, 0.0); +/// ctx.paint(); /// -/// // etc. +/// // 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 @@ -205,11 +224,9 @@ impl Context { /// /// ``` /// # fn main() -> Result<(), cairo::Error> { - /// use cairo::{Context, Format, ImageSurface}; - /// - /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); - /// let ctx = Context::new(&surface).unwrap(); - /// + /// # 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); /// @@ -349,11 +366,9 @@ impl Context { /// /// ``` /// # fn main() -> Result<(), cairo::Error> { - /// use cairo::{Context, Format, ImageSurface}; - /// - /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); - /// let ctx = Context::new(&surface).unwrap(); - /// + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; /// // paint the background black /// ctx.paint()?; /// @@ -381,11 +396,9 @@ impl Context { /// /// ``` /// # fn main() -> Result<(), cairo::Error> { - /// use cairo::{Context, Format, ImageSurface}; - /// - /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); - /// let ctx = Context::new(&surface).unwrap(); - /// + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; /// // paint the background black /// ctx.paint()?; /// @@ -419,12 +432,10 @@ impl Context { /// /// ``` /// # fn main() -> Result<(), cairo::Error> { - /// use cairo::{Context, Format, ImageSurface, RadialGradient}; - /// - /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); - /// let ctx = Context::new(&surface).unwrap(); - /// - /// // creates a radial gradient from red to blue + /// # 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 @@ -436,7 +447,7 @@ impl Context { /// // 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); /// - /// // paints the canvas with the gradient + /// // paint the canvas with the gradient /// ctx.set_source(&pattern)?; /// ctx.paint()?; /// # Ok(()) @@ -1086,11 +1097,9 @@ impl Context { /// /// ``` /// # fn main() -> Result<(), cairo::Error> { - /// use cairo::{Context, Format, ImageSurface}; - /// - /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); - /// let ctx = Context::new(&surface).unwrap(); - /// + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; /// // paint the background black /// ctx.paint()?; /// @@ -1159,11 +1168,9 @@ impl Context { /// /// ``` /// # fn main() -> Result<(), cairo::Error> { - /// use cairo::{Context, Format, ImageSurface}; - /// - /// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); - /// let ctx = Context::new(&surface).unwrap(); - /// + /// # use cairo::{Context, Format, ImageSurface}; + /// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; + /// # let ctx = Context::new(&surface)?; /// // paint the background black /// ctx.paint()?; /// diff --git a/cairo/src/paths.rs b/cairo/src/paths.rs index caa7ae47ad2a..e6106ea338dc 100644 --- a/cairo/src/paths.rs +++ b/cairo/src/paths.rs @@ -19,11 +19,9 @@ use crate::{ffi, PathDataType}; /// /// ``` /// # fn main() -> Result<(), cairo::Error> { -/// use cairo::{Context, Format, ImageSurface}; -/// -/// let surface = ImageSurface::create(Format::ARgb32, 100, 100).unwrap(); -/// let ctx = Context::new(&surface).unwrap(); -/// +/// # use cairo::{Context, Format, ImageSurface}; +/// # let surface = ImageSurface::create(Format::ARgb32, 100, 100)?; +/// # let ctx = Context::new(&surface)?; /// // paint the background black /// ctx.paint()?; ///