-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Add mutable string accessor macro. * Add an attachment loader wrapper. * temp patch spine_c to refcount obv temporary, see EsotericSoftware/spine-runtimes#2399 * Convenience fn for creating region attachments. * return nulerror * clippy * fix doc string refs * move unsafe to interior As far as I can tell, this is safe to do if the attachment is valid, once esoteric updates spine with the ref count increment in spSlot_setAttachment * undo * mark unsafe again * clippy * Update src/attachment_loader.rs * Update slot.rs --------- Co-authored-by: jabu <46233424+jabuwu@users.noreply.github.com>
- Loading branch information
1 parent
495a2f0
commit 2199add
Showing
5 changed files
with
201 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,129 @@ | ||
use crate::{ | ||
c::{ | ||
spAtlasAttachmentLoader_create, spAttachmentLoader, spAttachmentLoader_createAttachment, | ||
spAttachmentLoader_dispose, | ||
}, | ||
c_interface::{NewFromPtr, SyncPtr}, | ||
Atlas, Attachment, AttachmentType, RegionProps, Skin, | ||
}; | ||
|
||
/// Error types related to [`AttachmentLoader`](`crate::AttachmentLoader`). | ||
#[derive(Debug)] | ||
pub enum AttachmentLoaderError { | ||
/// Creating an attachment failed. | ||
/// Check [`error1`](`Self::error1`) and [`error2`](`Self::error2`) for more information. | ||
CreateAttachmentFailed, | ||
InvalidArgument { | ||
field: &'static str, | ||
}, | ||
} | ||
|
||
#[derive(Debug)] | ||
pub struct AttachmentLoader { | ||
c_attachment_loader: SyncPtr<spAttachmentLoader>, | ||
} | ||
|
||
impl NewFromPtr<spAttachmentLoader> for AttachmentLoader { | ||
unsafe fn new_from_ptr(c_attachment_loader: *mut spAttachmentLoader) -> Self { | ||
Self { | ||
c_attachment_loader: SyncPtr(c_attachment_loader), | ||
} | ||
} | ||
} | ||
|
||
impl AttachmentLoader { | ||
/// The spine runtime offers a default [`AttachmentLoader`](`crate::AttachmentLoader`) that | ||
/// loads attachments from an [`Atlas`](`crate::Atlas`). | ||
pub fn new_atlas_loader(atlas: &Atlas) -> Self { | ||
unsafe { | ||
let atlas_attachment_loader = spAtlasAttachmentLoader_create(atlas.c_ptr()); | ||
let attachment_loader = &mut (*atlas_attachment_loader).super_0; | ||
Self::new_from_ptr(attachment_loader) | ||
} | ||
} | ||
|
||
/// Creates an [`Attachment`](`crate::Attachment`) of a specified type. | ||
/// | ||
/// # Errors | ||
/// | ||
/// Returns [`AttachmentLoaderError::CreateAttachmentFailed`] if creating the attachment failed. | ||
/// Check [`error1`](`Self::error1`) and [`error2`](`Self::error2`) for more information. | ||
/// Returns [`AttachmentLoaderError::InvalidArgument`] if `name` or `path` contain a null byte. | ||
pub fn create_attachment( | ||
&self, | ||
skin: Option<Skin>, | ||
attachment_type: AttachmentType, | ||
name: &str, | ||
path: &str, | ||
) -> Result<Attachment, AttachmentLoaderError> { | ||
let c_name = std::ffi::CString::new(name) | ||
.map_err(|_| AttachmentLoaderError::InvalidArgument { field: "name" })?; | ||
let c_path = std::ffi::CString::new(path) | ||
.map_err(|_| AttachmentLoaderError::InvalidArgument { field: "path" })?; | ||
|
||
unsafe { | ||
let c_name = c_name.as_ptr(); | ||
let c_path = c_path.as_ptr(); | ||
let c_skin = skin.map_or(std::ptr::null_mut(), |skin| skin.c_ptr()); | ||
let c_sequence = std::ptr::null_mut(); // What is this for? | ||
|
||
let attachment = spAttachmentLoader_createAttachment( | ||
self.c_ptr(), | ||
c_skin, | ||
attachment_type as u32, | ||
c_name, | ||
c_path, | ||
c_sequence, | ||
); | ||
|
||
if attachment.is_null() { | ||
Err(AttachmentLoaderError::CreateAttachmentFailed) | ||
} else { | ||
Ok(Attachment::new_from_ptr(attachment)) | ||
} | ||
} | ||
} | ||
|
||
/// Convenience function for creating a [`RegionAttachment`](`crate::RegionAttachment`). | ||
/// | ||
/// # Errors | ||
/// | ||
/// Returns [`AttachmentLoaderError::CreateAttachmentFailed`] if creating the attachment failed. | ||
/// Check [`error1`](`Self::error1`) and [`error2`](`Self::error2`) for more information. | ||
/// Returns [`AttachmentLoaderError::InvalidArgument`] if `name` or `path` contain a null byte. | ||
pub fn create_region_attachment( | ||
&self, | ||
skin: Option<Skin>, | ||
name: &str, | ||
path: &str, | ||
props: &RegionProps, | ||
) -> Result<Attachment, AttachmentLoaderError> { | ||
let attachment = self.create_attachment(skin, AttachmentType::Region, name, path)?; | ||
|
||
let Some(mut region) = attachment.as_region() else { | ||
return Err(AttachmentLoaderError::CreateAttachmentFailed); | ||
}; | ||
|
||
region.update_from_props(props); | ||
|
||
Ok(attachment) | ||
} | ||
|
||
c_accessor_string!(error1, error1); | ||
c_accessor_string!(error2, error2); | ||
c_ptr!(c_attachment_loader, spAttachmentLoader); | ||
} | ||
|
||
impl Clone for AttachmentLoader { | ||
fn clone(&self) -> Self { | ||
unsafe { AttachmentLoader::new_from_ptr(self.c_ptr()) } | ||
} | ||
} | ||
|
||
impl Drop for AttachmentLoader { | ||
fn drop(&mut self) { | ||
unsafe { | ||
spAttachmentLoader_dispose(self.c_ptr()); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters