skia_safe/core/
path_measure.rsuse crate::{prelude::*, scalar, ContourMeasure, Matrix, Path, Point, Vector};
use skia_bindings::{self as sb, SkPathMeasure};
use std::fmt;
pub type PathMeasure = Handle<SkPathMeasure>;
impl NativeDrop for SkPathMeasure {
fn drop(&mut self) {
unsafe { sb::C_SkPathMeasure_destruct(self) }
}
}
bitflags! {
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct MatrixFlags : u32 {
const GET_POSITION = sb::SkPathMeasure_MatrixFlags_kGetPosition_MatrixFlag as _;
const GET_TANGENT = sb::SkPathMeasure_MatrixFlags_kGetTangent_MatrixFlag as _;
const GET_POS_AND_TAN = Self::GET_POSITION.bits() | Self::GET_TANGENT.bits();
}
}
impl Default for MatrixFlags {
fn default() -> Self {
Self::GET_POS_AND_TAN
}
}
impl Default for PathMeasure {
fn default() -> Self {
Self::from_native_c(unsafe { SkPathMeasure::new() })
}
}
impl fmt::Debug for PathMeasure {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("PathMeasure")
.field("current_measure", &self.current_measure())
.finish()
}
}
impl PathMeasure {
pub fn new(path: &Path, force_closed: bool, res_scale: impl Into<Option<scalar>>) -> Self {
Self::from_native_c(unsafe {
SkPathMeasure::new1(path.native(), force_closed, res_scale.into().unwrap_or(1.0))
})
}
#[deprecated(since = "0.48.0", note = "Use PathMeasure::new")]
pub fn from_path(
path: &Path,
force_closed: bool,
res_scale: impl Into<Option<scalar>>,
) -> Self {
Self::new(path, force_closed, res_scale)
}
pub fn set_path(&mut self, path: &Path, force_closed: bool) -> &mut Self {
unsafe { self.native_mut().setPath(path.native(), force_closed) }
self
}
pub fn length(&mut self) -> scalar {
unsafe { self.native_mut().getLength() }
}
#[must_use]
pub fn pos_tan(&mut self, distance: scalar) -> Option<(Point, Vector)> {
let mut position = Point::default();
let mut tangent = Vector::default();
unsafe {
self.native_mut()
.getPosTan(distance, position.native_mut(), tangent.native_mut())
}
.if_true_some((position, tangent))
}
#[must_use]
pub fn matrix(
&mut self,
distance: scalar,
flags: impl Into<Option<MatrixFlags>>,
) -> Option<Matrix> {
let mut m = Matrix::default();
unsafe {
self.native_mut().getMatrix(
distance,
m.native_mut(),
#[allow(clippy::useless_conversion)]
flags.into().unwrap_or_default().bits().try_into().unwrap(),
)
}
.if_true_some(m)
}
pub fn segment(
&mut self,
start_d: scalar,
stop_d: scalar,
start_with_move_to: bool,
) -> Option<Path> {
let mut p = Path::default();
unsafe {
self.native_mut()
.getSegment(start_d, stop_d, p.native_mut(), start_with_move_to)
}
.if_true_some(p)
}
#[allow(clippy::wrong_self_convention)]
pub fn is_closed(&mut self) -> bool {
unsafe { self.native_mut().isClosed() }
}
pub fn next_contour(&mut self) -> bool {
unsafe { self.native_mut().nextContour() }
}
pub fn current_measure(&self) -> &Option<ContourMeasure> {
ContourMeasure::from_unshared_ptr_ref(&self.native().fContour.fPtr)
}
}
#[cfg(test)]
mod tests {
use crate::{Path, PathMeasure, Point};
#[test]
fn current_measure() {
let mut path = Path::circle((0., 0.), 10.0, None);
path.add_path(
&Path::circle((100., 100.), 27.0, None),
Point::default(),
None,
);
let mut measure = PathMeasure::new(&path, false, None);
while measure.next_contour() {
eprintln!("contour: {:?}", measure.current_measure());
}
assert!(measure.current_measure().is_none());
}
}