#oop #interface #traits #cast #dynamic

nightly no-std dynamic-cast

The fifth pillar of OOP: dynamic casting

9 releases

0.2.0 Jun 5, 2025
0.1.7 Jun 5, 2025
0.1.6 May 30, 2025
0.1.5 Nov 11, 2024
0.1.3 Nov 9, 2023

#2425 in Rust patterns


Used in 3 crates

MIT/Apache

13KB
161 lines

maintenance: experimental

dynamic-cast

The fifth pillar of OOP: dynamic casting.

(If you are looking for the rest, check out the basic-oop crate.)

use dynamic_cast::{SupportsInterfaces, impl_supports_interfaces, dyn_cast_arc};
use std::sync::Arc;

// class Base

#[portrait::make]
trait IsBase: SupportsInterfaces {
    fn base_method(&self) -> u8;
}

struct Base {
    data: u8,
}

impl_supports_interfaces!(Base: IsBase);

impl IsBase for Base {
    fn base_method(&self) -> u8 { self.data }
}

// class Descendant: Base

#[portrait::make]
trait IsDescendant: IsBase + SupportsInterfaces {
    fn descendant_method(&self) -> u8;
}

struct Descendant {
    base: Base,
    data: u8,
}

impl_supports_interfaces!(Descendant: IsBase, IsDescendant);

#[portrait::fill(portrait::delegate(Base; self.base))]
impl IsBase for Descendant { }

impl IsDescendant for Descendant {
    fn descendant_method(&self) -> u8 { self.data }
}

// casting

fn main() {
    let a: Arc<dyn IsDescendant> = 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 IsBase> = dyn_cast_arc(a).unwrap();
    assert_eq!(a_as_base.base_method(), 1);
    let a_as_descendant: Arc<dyn IsDescendant> = dyn_cast_arc(a_as_base).unwrap();
    assert_eq!(a_as_descendant.descendant_method(), 2);
    assert_eq!(a_as_descendant.base_method(), 1);
}

Dependencies

~43KB