use {
crate::{
ffi::graphics as ffi,
graphics::{
Color, Drawable, FloatRect, IntRect, RenderStates, RenderTarget, Shape, Texture,
Transform, Transformable,
},
system::Vector2f,
},
std::{
marker::PhantomData,
ptr::{self, NonNull},
},
};
#[derive(Debug)]
pub struct ConvexShape<'s> {
handle: NonNull<ffi::sfConvexShape>,
texture: PhantomData<&'s Texture>,
}
impl<'s> ConvexShape<'s> {
#[must_use]
pub fn new(points_count: usize) -> ConvexShape<'s> {
let shape = unsafe { ffi::sfConvexShape_new() };
let mut shape = ConvexShape {
handle: NonNull::new(shape).expect("Failed to create ConvexShape"),
texture: PhantomData,
};
shape.set_point_count(points_count);
shape
}
#[must_use]
pub fn with_texture(points_count: usize, texture: &'s Texture) -> ConvexShape<'s> {
let mut shape = ConvexShape::new(points_count);
shape.set_texture(texture, true);
shape
}
pub fn set_point<P: Into<Vector2f>>(&mut self, index: usize, point: P) {
assert!(
index < self.point_count(),
"Index out of bounds. Index: {}, len: {}",
index,
self.point_count()
);
unsafe { ffi::sfConvexShape_setPoint(self.handle.as_ptr(), index, point.into()) }
}
pub fn set_point_count(&mut self, count: usize) {
unsafe { ffi::sfConvexShape_setPointCount(self.handle.as_ptr(), count) }
}
pub(super) fn raw(&self) -> *const ffi::sfConvexShape {
self.handle.as_ptr()
}
}
impl Drawable for ConvexShape<'_> {
fn draw<'a: 'shader, 'texture, 'shader, 'shader_texture>(
&'a self,
target: &mut dyn RenderTarget,
states: &RenderStates<'texture, 'shader, 'shader_texture>,
) {
target.draw_convex_shape(self, states)
}
}
impl Transformable for ConvexShape<'_> {
fn set_position<P: Into<Vector2f>>(&mut self, position: P) {
unsafe { ffi::sfConvexShape_setPosition(self.handle.as_ptr(), position.into()) }
}
fn set_rotation(&mut self, angle: f32) {
unsafe { ffi::sfConvexShape_setRotation(self.handle.as_ptr(), angle) }
}
fn set_scale<S: Into<Vector2f>>(&mut self, scale: S) {
unsafe { ffi::sfConvexShape_setScale(self.handle.as_ptr(), scale.into()) }
}
fn set_origin<O: Into<Vector2f>>(&mut self, origin: O) {
unsafe { ffi::sfConvexShape_setOrigin(self.handle.as_ptr(), origin.into()) }
}
fn position(&self) -> Vector2f {
unsafe { ffi::sfConvexShape_getPosition(self.handle.as_ptr()) }
}
fn rotation(&self) -> f32 {
unsafe { ffi::sfConvexShape_getRotation(self.handle.as_ptr()) }
}
fn get_scale(&self) -> Vector2f {
unsafe { ffi::sfConvexShape_getScale(self.handle.as_ptr()) }
}
fn origin(&self) -> Vector2f {
unsafe { ffi::sfConvexShape_getOrigin(self.handle.as_ptr()) }
}
fn move_<O: Into<Vector2f>>(&mut self, offset: O) {
unsafe { ffi::sfConvexShape_move(self.handle.as_ptr(), offset.into()) }
}
fn rotate(&mut self, angle: f32) {
unsafe { ffi::sfConvexShape_rotate(self.handle.as_ptr(), angle) }
}
fn scale<F: Into<Vector2f>>(&mut self, factors: F) {
unsafe { ffi::sfConvexShape_scale(self.handle.as_ptr(), factors.into()) }
}
fn transform(&self) -> &Transform {
unsafe { &*ffi::sfConvexShape_getTransform(self.handle.as_ptr()) }
}
fn inverse_transform(&self) -> &Transform {
unsafe { &*ffi::sfConvexShape_getInverseTransform(self.handle.as_ptr()) }
}
}
impl<'s> Shape<'s> for ConvexShape<'s> {
fn set_texture(&mut self, texture: &'s Texture, reset_rect: bool) {
unsafe { ffi::sfConvexShape_setTexture(self.handle.as_ptr(), texture, reset_rect) }
}
fn disable_texture(&mut self) {
unsafe { ffi::sfConvexShape_setTexture(self.handle.as_ptr(), ptr::null_mut(), true) }
}
fn set_texture_rect(&mut self, rect: IntRect) {
unsafe { ffi::sfConvexShape_setTextureRect(self.handle.as_ptr(), rect) }
}
fn set_fill_color(&mut self, color: Color) {
unsafe { ffi::sfConvexShape_setFillColor(self.handle.as_ptr(), color) }
}
fn set_outline_color(&mut self, color: Color) {
unsafe { ffi::sfConvexShape_setOutlineColor(self.handle.as_ptr(), color) }
}
fn set_outline_thickness(&mut self, thickness: f32) {
unsafe { ffi::sfConvexShape_setOutlineThickness(self.handle.as_ptr(), thickness) }
}
fn texture(&self) -> Option<&'s Texture> {
unsafe { ffi::sfConvexShape_getTexture(self.handle.as_ptr()).as_ref() }
}
fn texture_rect(&self) -> IntRect {
unsafe { ffi::sfConvexShape_getTextureRect(self.handle.as_ptr()) }
}
fn fill_color(&self) -> Color {
unsafe { ffi::sfConvexShape_getFillColor(self.handle.as_ptr()) }
}
fn outline_color(&self) -> Color {
unsafe { ffi::sfConvexShape_getOutlineColor(self.handle.as_ptr()) }
}
fn outline_thickness(&self) -> f32 {
unsafe { ffi::sfConvexShape_getOutlineThickness(self.handle.as_ptr()) }
}
fn point_count(&self) -> usize {
unsafe { ffi::sfConvexShape_getPointCount(self.handle.as_ptr()) }
}
fn point(&self, index: usize) -> Vector2f {
unsafe {
assert!(
index < self.point_count(),
"Index out of bounds. Index: {}, len: {}",
index,
self.point_count()
);
ffi::sfConvexShape_getPoint(self.handle.as_ptr(), index)
}
}
fn local_bounds(&self) -> FloatRect {
unsafe { ffi::sfConvexShape_getLocalBounds(self.handle.as_ptr()) }
}
fn global_bounds(&self) -> FloatRect {
unsafe { ffi::sfConvexShape_getGlobalBounds(self.handle.as_ptr()) }
}
}
impl<'s> Clone for ConvexShape<'s> {
fn clone(&self) -> ConvexShape<'s> {
let shape = unsafe { ffi::sfConvexShape_cpy(self.handle.as_ptr()) };
ConvexShape {
handle: NonNull::new(shape).expect("Not enough memory to clone ConvexShape"),
texture: self.texture,
}
}
}
impl Drop for ConvexShape<'_> {
fn drop(&mut self) {
unsafe { ffi::sfConvexShape_del(self.handle.as_ptr()) }
}
}