gsk4/auto/path.rs
1// This file was generated by gir (https://github.com/gtk-rs/gir)
2// from gir-files (https://github.com/gtk-rs/gir-files)
3// DO NOT EDIT
4
5use crate::{FillRule, PathPoint, Stroke, ffi};
6use glib::translate::*;
7
8glib::wrapper! {
9 /// Describes lines and curves that are more complex than simple rectangles.
10 ///
11 /// Paths can used for rendering (filling or stroking) and for animations
12 /// (e.g. as trajectories).
13 ///
14 /// [`Path`][crate::Path] is an immutable, opaque, reference-counted struct.
15 /// After creation, you cannot change the types it represents. Instead,
16 /// new [`Path`][crate::Path] objects have to be created. The [`PathBuilder`][crate::PathBuilder]
17 /// structure is meant to help in this endeavor.
18 ///
19 /// Conceptually, a path consists of zero or more contours (continuous, connected
20 /// curves), each of which may or may not be closed. Contours are typically
21 /// constructed from Bézier segments.
22 ///
23 /// <picture>
24 /// <source srcset="path-dark.png" media="(prefers-color-scheme: dark)">
25 /// <img alt="A Path" src="path-light.png">
26 /// </picture>
27 #[derive(Debug)]
28 pub struct Path(Shared<ffi::GskPath>);
29
30 match fn {
31 ref => |ptr| ffi::gsk_path_ref(ptr),
32 unref => |ptr| ffi::gsk_path_unref(ptr),
33 type_ => || ffi::gsk_path_get_type(),
34 }
35}
36
37impl Path {
38 #[cfg(feature = "v4_22")]
39 #[cfg_attr(docsrs, doc(cfg(feature = "v4_22")))]
40 #[doc(alias = "gsk_path_equal")]
41 fn equal(&self, path2: &Path) -> bool {
42 unsafe {
43 from_glib(ffi::gsk_path_equal(
44 self.to_glib_none().0,
45 path2.to_glib_none().0,
46 ))
47 }
48 }
49
50 /// Computes the bounds of the given path.
51 ///
52 /// The returned bounds may be larger than necessary, because this
53 /// function aims to be fast, not accurate. The bounds are guaranteed
54 /// to contain the path. For accurate bounds, use
55 /// [`tight_bounds()`][Self::tight_bounds()].
56 ///
57 /// It is possible that the returned rectangle has 0 width and/or height.
58 /// This can happen when the path only describes a point or an
59 /// axis-aligned line.
60 ///
61 /// If the path is empty, false is returned and @bounds are set to
62 /// graphene_rect_zero(). This is different from the case where the path
63 /// is a single point at the origin, where the @bounds will also be set to
64 /// the zero rectangle but true will be returned.
65 ///
66 /// # Returns
67 ///
68 /// true if the path has bounds, false if the path is known
69 /// to be empty and have no bounds
70 ///
71 /// ## `bounds`
72 /// return location for the bounds
73 #[doc(alias = "gsk_path_get_bounds")]
74 #[doc(alias = "get_bounds")]
75 pub fn bounds(&self) -> Option<graphene::Rect> {
76 unsafe {
77 let mut bounds = graphene::Rect::uninitialized();
78 let ret = from_glib(ffi::gsk_path_get_bounds(
79 self.to_glib_none().0,
80 bounds.to_glib_none_mut().0,
81 ));
82 if ret { Some(bounds) } else { None }
83 }
84 }
85
86 /// Computes the closest point on the path to the given point.
87 ///
88 /// If there is no point closer than the given threshold,
89 /// false is returned.
90 /// ## `point`
91 /// the point
92 /// ## `threshold`
93 /// maximum allowed distance
94 ///
95 /// # Returns
96 ///
97 /// true if @point was set to the closest point
98 /// on @self, false if no point is closer than @threshold
99 ///
100 /// ## `result`
101 /// return location for the closest point
102 ///
103 /// ## `distance`
104 /// return location for the distance
105 #[doc(alias = "gsk_path_get_closest_point")]
106 #[doc(alias = "get_closest_point")]
107 pub fn closest_point(
108 &self,
109 point: &graphene::Point,
110 threshold: f32,
111 ) -> Option<(PathPoint, f32)> {
112 unsafe {
113 let mut result = PathPoint::uninitialized();
114 let mut distance = std::mem::MaybeUninit::uninit();
115 let ret = from_glib(ffi::gsk_path_get_closest_point(
116 self.to_glib_none().0,
117 point.to_glib_none().0,
118 threshold,
119 result.to_glib_none_mut().0,
120 distance.as_mut_ptr(),
121 ));
122 if ret {
123 Some((result, distance.assume_init()))
124 } else {
125 None
126 }
127 }
128 }
129
130 /// Gets the end point of the path.
131 ///
132 /// An empty path has no points, so false
133 /// is returned in this case.
134 ///
135 /// # Returns
136 ///
137 /// true if @result was filled
138 ///
139 /// ## `result`
140 /// return location for point
141 #[doc(alias = "gsk_path_get_end_point")]
142 #[doc(alias = "get_end_point")]
143 pub fn end_point(&self) -> Option<PathPoint> {
144 unsafe {
145 let mut result = PathPoint::uninitialized();
146 let ret = from_glib(ffi::gsk_path_get_end_point(
147 self.to_glib_none().0,
148 result.to_glib_none_mut().0,
149 ));
150 if ret { Some(result) } else { None }
151 }
152 }
153
154 //#[cfg(feature = "v4_22")]
155 //#[cfg_attr(docsrs, doc(cfg(feature = "v4_22")))]
156 //#[doc(alias = "gsk_path_get_next")]
157 //#[doc(alias = "get_next")]
158 //pub fn next(&self, point: /*Unimplemented*/PathPoint) -> bool {
159 // unsafe { TODO: call ffi:gsk_path_get_next() }
160 //}
161
162 //#[cfg(feature = "v4_22")]
163 //#[cfg_attr(docsrs, doc(cfg(feature = "v4_22")))]
164 //#[doc(alias = "gsk_path_get_previous")]
165 //#[doc(alias = "get_previous")]
166 //pub fn previous(&self, point: /*Unimplemented*/PathPoint) -> bool {
167 // unsafe { TODO: call ffi:gsk_path_get_previous() }
168 //}
169
170 /// Gets the start point of the path.
171 ///
172 /// An empty path has no points, so false
173 /// is returned in this case.
174 ///
175 /// # Returns
176 ///
177 /// true if @result was filled
178 ///
179 /// ## `result`
180 /// return location for point
181 #[doc(alias = "gsk_path_get_start_point")]
182 #[doc(alias = "get_start_point")]
183 pub fn start_point(&self) -> Option<PathPoint> {
184 unsafe {
185 let mut result = PathPoint::uninitialized();
186 let ret = from_glib(ffi::gsk_path_get_start_point(
187 self.to_glib_none().0,
188 result.to_glib_none_mut().0,
189 ));
190 if ret { Some(result) } else { None }
191 }
192 }
193
194 /// Computes the bounds for stroking the given path with the
195 /// given parameters.
196 ///
197 /// The returned bounds may be larger than necessary, because this
198 /// function aims to be fast, not accurate. The bounds are guaranteed
199 /// to contain the area affected by the stroke, including protrusions
200 /// like miters.
201 /// ## `stroke`
202 /// stroke parameters
203 ///
204 /// # Returns
205 ///
206 /// true if the path has bounds, false if the path is known
207 /// to be empty and have no bounds.
208 ///
209 /// ## `bounds`
210 /// the bounds to fill in
211 #[doc(alias = "gsk_path_get_stroke_bounds")]
212 #[doc(alias = "get_stroke_bounds")]
213 pub fn stroke_bounds(&self, stroke: &Stroke) -> Option<graphene::Rect> {
214 unsafe {
215 let mut bounds = graphene::Rect::uninitialized();
216 let ret = from_glib(ffi::gsk_path_get_stroke_bounds(
217 self.to_glib_none().0,
218 stroke.to_glib_none().0,
219 bounds.to_glib_none_mut().0,
220 ));
221 if ret { Some(bounds) } else { None }
222 }
223 }
224
225 /// Computes the tight bounds of the given path.
226 ///
227 /// This function works harder than [`bounds()`][Self::bounds()] to
228 /// produce the smallest possible bounds.
229 ///
230 /// # Returns
231 ///
232 /// true if the path has bounds, false if the path is known
233 /// to be empty and have no bounds
234 ///
235 /// ## `bounds`
236 /// return location for the bounds
237 #[cfg(feature = "v4_22")]
238 #[cfg_attr(docsrs, doc(cfg(feature = "v4_22")))]
239 #[doc(alias = "gsk_path_get_tight_bounds")]
240 #[doc(alias = "get_tight_bounds")]
241 pub fn tight_bounds(&self) -> Option<graphene::Rect> {
242 unsafe {
243 let mut bounds = graphene::Rect::uninitialized();
244 let ret = from_glib(ffi::gsk_path_get_tight_bounds(
245 self.to_glib_none().0,
246 bounds.to_glib_none_mut().0,
247 ));
248 if ret { Some(bounds) } else { None }
249 }
250 }
251
252 /// Returns whether a point is inside the fill area of a path.
253 ///
254 /// Note that this function assumes that filling a contour
255 /// implicitly closes it.
256 /// ## `point`
257 /// the point to test
258 /// ## `fill_rule`
259 /// the fill rule to follow
260 ///
261 /// # Returns
262 ///
263 /// true if @point is inside
264 #[doc(alias = "gsk_path_in_fill")]
265 pub fn in_fill(&self, point: &graphene::Point, fill_rule: FillRule) -> bool {
266 unsafe {
267 from_glib(ffi::gsk_path_in_fill(
268 self.to_glib_none().0,
269 point.to_glib_none().0,
270 fill_rule.into_glib(),
271 ))
272 }
273 }
274
275 /// Returns if the path represents a single closed contour.
276 ///
277 /// # Returns
278 ///
279 /// true if the path is closed
280 #[doc(alias = "gsk_path_is_closed")]
281 pub fn is_closed(&self) -> bool {
282 unsafe { from_glib(ffi::gsk_path_is_closed(self.to_glib_none().0)) }
283 }
284
285 /// Checks if the path is empty, i.e. contains no lines or curves.
286 ///
287 /// # Returns
288 ///
289 /// true if the path is empty
290 #[doc(alias = "gsk_path_is_empty")]
291 pub fn is_empty(&self) -> bool {
292 unsafe { from_glib(ffi::gsk_path_is_empty(self.to_glib_none().0)) }
293 }
294
295 /// Appends the path to a cairo context for drawing with Cairo.
296 ///
297 /// This may cause some suboptimal conversions to be performed as
298 /// Cairo does not support all features of [`Path`][crate::Path].
299 ///
300 /// This function does not clear the existing Cairo path. Call
301 /// cairo_new_path() if you want this.
302 /// ## `cr`
303 /// a cairo context
304 #[doc(alias = "gsk_path_to_cairo")]
305 pub fn to_cairo(&self, cr: &cairo::Context) {
306 unsafe {
307 ffi::gsk_path_to_cairo(self.to_glib_none().0, mut_override(cr.to_glib_none().0));
308 }
309 }
310
311 /// Converts the path into a human-readable string.
312 ///
313 /// You can use this function in a debugger to get a quick overview
314 /// of the path.
315 ///
316 /// This is a wrapper around `Gsk::Path::print()`, see that function
317 /// for details.
318 ///
319 /// # Returns
320 ///
321 /// a new string for @self
322 #[doc(alias = "gsk_path_to_string")]
323 #[doc(alias = "to_string")]
324 pub fn to_str(&self) -> glib::GString {
325 unsafe { from_glib_full(ffi::gsk_path_to_string(self.to_glib_none().0)) }
326 }
327
328 /// Constructs a path from a serialized form.
329 ///
330 /// The string is expected to be in (a superset of)
331 /// [SVG path syntax](https://www.w3.org/TR/SVG11/paths.html#PathData),
332 /// as e.g. produced by [`to_str()`][Self::to_str()].
333 ///
334 /// A high-level summary of the syntax:
335 ///
336 /// - `M x y` Move to `(x, y)`
337 /// - `L x y` Add a line from the current point to `(x, y)`
338 /// - `Q x1 y1 x2 y2` Add a quadratic Bézier from the current point to `(x2, y2)`, with control point `(x1, y1)`
339 /// - `C x1 y1 x2 y2 x3 y3` Add a cubic Bézier from the current point to `(x3, y3)`, with control points `(x1, y1)` and `(x2, y2)`
340 /// - `Z` Close the contour by drawing a line back to the start point
341 /// - `H x` Add a horizontal line from the current point to the given x value
342 /// - `V y` Add a vertical line from the current point to the given y value
343 /// - `T x2 y2` Add a quadratic Bézier, using the reflection of the previous segments' control point as control point
344 /// - `S x2 y2 x3 y3` Add a cubic Bézier, using the reflection of the previous segments' second control point as first control point
345 /// - `A rx ry r l s x y` Add an elliptical arc from the current point to `(x, y)` with radii rx and ry. See the SVG documentation for how the other parameters influence the arc.
346 /// - `O x1 y1 x2 y2 w` Add a rational quadratic Bézier from the current point to `(x2, y2)` with control point `(x1, y1)` and weight `w`.
347 ///
348 /// All the commands have lowercase variants that interpret coordinates
349 /// relative to the current point.
350 ///
351 /// The `O` command is an extension that is not supported in SVG.
352 /// ## `string`
353 /// a string
354 ///
355 /// # Returns
356 ///
357 /// a new [`Path`][crate::Path], or `NULL` if @string could not be parsed
358 #[doc(alias = "gsk_path_parse")]
359 pub fn parse(string: &str) -> Result<Path, glib::BoolError> {
360 assert_initialized_main_thread!();
361 unsafe {
362 Option::<_>::from_glib_full(ffi::gsk_path_parse(string.to_glib_none().0))
363 .ok_or_else(|| glib::bool_error!("Can't parse Path"))
364 }
365 }
366}
367
368#[cfg(feature = "v4_22")]
369#[cfg_attr(docsrs, doc(cfg(feature = "v4_22")))]
370impl PartialEq for Path {
371 #[inline]
372 fn eq(&self, other: &Self) -> bool {
373 self.equal(other)
374 }
375}
376
377#[cfg(feature = "v4_22")]
378#[cfg_attr(docsrs, doc(cfg(feature = "v4_22")))]
379impl Eq for Path {}
380
381impl std::fmt::Display for Path {
382 #[inline]
383 fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
384 f.write_str(&self.to_str())
385 }
386}