The fifth pillar of OOP: dynamic casting.
use dynamic_cast::{SupportsInterfaces, impl_supports_interfaces, dyn_cast_arc};
use std::sync::Arc;
// class Base
#[portrait::make]
trait TBase: SupportsInterfaces {
fn base_method(&self) -> u8;
}
struct Base {
data: u8,
}
impl_supports_interfaces!(Base: TBase);
impl TBase for Base {
fn base_method(&self) -> u8 { self.data }
}
// class Descendant: Base
#[portrait::make]
trait TDescendant: TBase + SupportsInterfaces {
fn descendant_method(&self) -> u8;
}
struct Descendant {
base: Base,
data: u8,
}
impl_supports_interfaces!(Descendant: TBase, TDescendant);
#[portrait::fill(portrait::delegate(Base; self.base))]
impl TBase for Descendant { }
impl TDescendant for Descendant {
fn descendant_method(&self) -> u8 { self.data }
}
// casting
fn main() {
let a: Arc<dyn TDescendant> = Arc::new(Descendant { base: Base { data: 1 }, data: 2 });
assert_eq!(a.descendant_method(), 2);
assert_eq!(a.base_method(), 1);
let a_as_base: Arc<dyn TBase> = dyn_cast_arc(a).unwrap();
assert_eq!(a_as_base.base_method(), 1);
let a_as_descendant: Arc<dyn TDescendant> = dyn_cast_arc(a_as_base).unwrap();
assert_eq!(a_as_descendant.descendant_method(), 2);
assert_eq!(a_as_descendant.base_method(), 1);
}