use core::ffi::c_char;
use core::ptr;
use std::ffi::CString;
use std::path::Path;
use crate::error::{from_swift, VisionError};
use crate::ffi;
use crate::recognize_text::BoundingBox;
#[derive(Debug, Clone, PartialEq)]
pub struct DetectedFace {
pub bounding_box: BoundingBox,
pub confidence: f32,
pub roll: Option<f32>,
pub yaw: Option<f32>,
pub pitch: Option<f32>,
}
#[derive(Debug, Clone, Copy, Default)]
pub struct FaceDetector;
impl FaceDetector {
#[must_use]
pub const fn new() -> Self {
Self
}
pub fn detect_in_path(&self, path: impl AsRef<Path>) -> Result<Vec<DetectedFace>, VisionError> {
let path_str = path
.as_ref()
.to_str()
.ok_or_else(|| VisionError::InvalidArgument("non-UTF-8 path".into()))?;
let path_c = CString::new(path_str)
.map_err(|e| VisionError::InvalidArgument(format!("path NUL byte: {e}")))?;
let mut out_array: *mut core::ffi::c_void = ptr::null_mut();
let mut out_count: usize = 0;
let mut err_msg: *mut c_char = ptr::null_mut();
let status = unsafe {
ffi::vn_detect_faces_in_path(
path_c.as_ptr(),
&mut out_array,
&mut out_count,
&mut err_msg,
)
};
if status != ffi::status::OK {
return Err(unsafe { from_swift(status, err_msg) });
}
Self::collect(out_array, out_count)
}
pub fn detect_in_pixel_buffer(
&self,
pixel_buffer: &apple_cf::cv::CVPixelBuffer,
) -> Result<Vec<DetectedFace>, VisionError> {
let mut out_array: *mut core::ffi::c_void = ptr::null_mut();
let mut out_count: usize = 0;
let mut err_msg: *mut c_char = ptr::null_mut();
let status = unsafe {
ffi::vn_detect_faces_in_pixel_buffer(
pixel_buffer.as_ptr(),
&mut out_array,
&mut out_count,
&mut err_msg,
)
};
if status != ffi::status::OK {
return Err(unsafe { from_swift(status, err_msg) });
}
Self::collect(out_array, out_count)
}
#[allow(clippy::unnecessary_wraps)]
fn collect(
out_array: *mut core::ffi::c_void,
out_count: usize,
) -> Result<Vec<DetectedFace>, VisionError> {
if out_array.is_null() || out_count == 0 {
return Ok(Vec::new());
}
let typed_array = out_array.cast::<ffi::DetectedFaceRaw>();
let mut results = Vec::with_capacity(out_count);
for i in 0..out_count {
let raw = unsafe { &*typed_array.add(i) };
let nan_to_none = |v: f32| if v.is_nan() { None } else { Some(v) };
results.push(DetectedFace {
bounding_box: BoundingBox {
x: raw.bbox_x,
y: raw.bbox_y,
width: raw.bbox_w,
height: raw.bbox_h,
},
confidence: raw.confidence,
roll: nan_to_none(raw.roll),
yaw: nan_to_none(raw.yaw),
pitch: nan_to_none(raw.pitch),
});
}
unsafe { ffi::vn_detected_faces_free(out_array, out_count) };
Ok(results)
}
}