mod frame_impl;
mod kind;
use alloc::boxed::Box;
#[cfg(nightly)]
use core::any::{self, Demand, Provider};
use core::{any::TypeId, fmt, panic::Location};
use self::frame_impl::FrameImpl;
pub use self::kind::{AttachmentKind, FrameKind};
pub struct Frame {
frame: Box<dyn FrameImpl>,
location: &'static Location<'static>,
sources: Box<[Frame]>,
}
impl Frame {
#[must_use]
pub const fn location(&self) -> &'static Location<'static> {
self.location
}
#[allow(missing_docs)]
#[must_use]
#[deprecated(since = "0.2.0", note = "use `sources()` instead")]
pub const fn source(&self) -> Option<&Self> {
self.sources().first()
}
#[must_use]
pub const fn sources(&self) -> &[Self] {
&self.sources
}
#[allow(missing_docs)]
#[must_use]
#[deprecated(since = "0.2.0", note = "use `sources_mut()` instead")]
pub fn source_mut(&mut self) -> Option<&mut Self> {
self.sources_mut().first_mut()
}
#[must_use]
pub fn sources_mut(&mut self) -> &mut [Self] {
&mut self.sources
}
#[must_use]
pub fn kind(&self) -> FrameKind<'_> {
self.frame.kind()
}
#[must_use]
#[cfg(nightly)]
pub fn request_ref<T>(&self) -> Option<&T>
where
T: ?Sized + 'static,
{
any::request_ref(self)
}
#[must_use]
#[cfg(nightly)]
pub fn request_value<T>(&self) -> Option<T>
where
T: 'static,
{
any::request_value(self)
}
#[must_use]
pub fn is<T: Send + Sync + 'static>(&self) -> bool {
self.downcast_ref::<T>().is_some()
}
#[must_use]
pub fn downcast_ref<T: Send + Sync + 'static>(&self) -> Option<&T> {
(TypeId::of::<T>() == Self::type_id(self)).then(|| {
unsafe { &*(self.frame.as_ref() as *const dyn FrameImpl).cast::<T>() }
})
}
#[must_use]
pub fn downcast_mut<T: Send + Sync + 'static>(&mut self) -> Option<&mut T> {
(TypeId::of::<T>() == Self::type_id(self)).then(|| {
unsafe { &mut *(self.frame.as_mut() as *mut dyn FrameImpl).cast::<T>() }
})
}
pub(crate) fn type_id(&self) -> TypeId {
FrameImpl::type_id(&*self.frame)
}
}
#[cfg(nightly)]
impl Provider for Frame {
fn provide<'a>(&'a self, demand: &mut Demand<'a>) {
self.frame.provide(demand);
}
}
impl fmt::Debug for Frame {
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
let mut debug = fmt.debug_struct("Frame");
debug.field("location", self.location());
match self.kind() {
FrameKind::Context(context) => {
debug.field("context", &context);
}
FrameKind::Attachment(AttachmentKind::Printable(attachment)) => {
debug.field("attachment", &attachment);
}
FrameKind::Attachment(AttachmentKind::Opaque(_)) => {
debug.field("attachment", &"opaque");
}
}
debug.finish()
}
}