evtc/extract.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48
//! Interfaces for extracting information from events.
use crate::Event;
/// Forcefully extracts information from a combat event.
pub trait Extract: Sized {
/// Extracts [`Self`] from the combat event.
///
/// # Safety
/// This is safe when the given event is a valid event to extract [`Self`] from.
unsafe fn extract(event: &Event) -> Self;
}
/// Attempts to extract information from a combat event.
pub trait TryExtract: Sized + Extract {
/// Attempts to extract [`Self`] from the combat event.
#[inline]
fn try_extract(event: &Event) -> Option<Self> {
Self::can_extract(event).then(|| unsafe { Self::extract(event) })
}
/// Checks whether [`Self`] can be extracted from the event.
fn can_extract(event: &Event) -> bool;
}
/// Helper to transmute [`Event`] fields.
///
/// # Usage
/// ```ignore
/// let value = transmute_field!(event.src_agent as [f32; 3]);
/// ```
macro_rules! transmute_field {
( $event: ident . $field: ident as $ty: tt ) => {{
const _: () = {
let end = ::memoffset::offset_of!($crate::Event, $field) + ::std::mem::size_of::<$ty>();
assert!(
end <= ::std::mem::size_of::<$crate::Event>(),
"transmute field outside of event",
);
};
let event: &$crate::Event = ::std::borrow::Borrow::borrow($event);
let field_ptr = ::std::ptr::addr_of!(event.$field);
(*field_ptr.cast::<$ty>()).clone()
}};
}
pub(crate) use transmute_field;