Callbacks, Trait Objects & Associated Types, Oh My!

Ben Lovy - Sep 3 '19 - - Dev Community

Last week, I promised a dive into Rust procedural macros as I work to design a DSL for my Widget UI building-block trait. What I've learned is not to ever promise things like that, because I didn't even touch that aspect this week. You may want to scooch back off the edge of your seat for at least another week.

Instead, my Widgets are clickable now! A Button you can't click is pretty useless, so I figured I'd get that system functional before trying anything fancy on top of it.

Callbacks

In order to keep the rendering stuff decoupled from the game logic, I need users of this library to be able to define clickable functionality however they please. The solution I settled on is a little bit Flux-ish - to create a clickable widget, you provide a callback that returns an action type. When a click is registered, the grid dives through its children to see where exactly the click was located. If a match is found, that action will bubble up through successive handle_click calls until some parent widget above it decides to handle that action.

This affords a pretty high degree of flexibility - any widget in your application can choose to handle a message. This allows you to use container widgets that can handle their children as a group, and only pass up what's necessary for a global state change if needed.

The first problem was representing this callback in a way that's clone-able and easy to store in a struct. I ended up using an Rc, or reference-counted smart pointer:

pub struct Callback<T> {
    f: Rc<dyn Fn() -> T>,
}

impl<T> Callback<T> {
    /// Call this callback
    pub fn call(&self) -> T {
        (self.f)()
    }
}

impl<T> Clone for Callback<T> {
    fn clone(&self) -> Self {
        Self {
            f: Rc::clone(&self.f),
        }
    }
}
Enter fullscreen mode Exit fullscreen mode

To construct them, you can call Callback::from() on a closure, as used in the demo app:

let button = Button::new(
    // Display name
    &format!("{:?}", self.value),
    // Optional size, if None will calculate based on display name
    Some((VALUES.die_dimension, VALUES.die_dimension).into()),
    // Border color
    Color::from_str("black").unwrap(),
    // Click action
    Some(Callback::from(move || -> FiveDiceMessage {
        FiveDiceMessage::HoldDie(id)
    })),
);
Enter fullscreen mode Exit fullscreen mode

The caveat is that I haven't figured out how not to require that the return type have a 'static lifetime:

impl<T, F: Fn() -> T + 'static> From<F> for Callback<T> {
    fn from(func: F) -> Self {
        Self { f: Rc::new(func) }
    }
}
Enter fullscreen mode Exit fullscreen mode

This is part of what led me towards the action-reducer type thing - the actions themselves can be just plain data like the example! You define an enum for your messages:

#[derive(Debug, Clone, Copy)]
pub enum FiveDiceMessage {
    HoldDie(usize),
    RollDice,
    StartOver,
}
Enter fullscreen mode Exit fullscreen mode

And then a reducer to handle them:

impl Game {
    // ..

    /// Handle all incoming messages
    fn reducer(&mut self, msg: FiveDiceMessage) {
        use FiveDiceMessage::*;
        match msg {
            HoldDie(idx) => self.hold_die(idx),
            RollDice => self.roll_dice(),
            StartOver => self.reset(),
        }
    }

    // ..
}
Enter fullscreen mode Exit fullscreen mode

It's not a real "reducer", we're mutating in place instead of using pure functions, but it's a similar pattern.

Ideally, it'd be nice to be able to accept more generic callbacks, and not lock users into this pattern. I'm good with this for now though. However, It was a little bit tricky integrating this into my Widget definition. In the previous post, I gave the trait definitions I was working with:

/// Trait representing things that can be drawn to the canvas
pub trait Drawable {
    /// Draw this game element with the given top left corner
    /// Only ever called once mounted.  Returns the bottom right corner of what was painted
    fn draw_at(&self, top_left: Point, w: WindowPtr) -> Result<Point>;
    /// Get the Region of the bounding box of this drawable
    fn get_region(&self, top_left: Point, w: WindowPtr) -> Result<Region>;
}

/// Trait representing sets of 0 or more Drawables
/// Each one can have variable number rows and elements in each row
pub trait Widget {
    /// Get the total of all regions of this widget
    fn get_region(&self, top_left: Point, w: WindowPtr) -> Result<Region>;
    /// Make this object into a Widget.  Takes an optional callback
    fn mount_widget(&self) -> MountedWidget;
}
Enter fullscreen mode Exit fullscreen mode

I need to add a method to Widget that detects a click and bubbles up whatever the callback returns:

pub trait Widget {
    // ..
    /// Handle a click in this region
    fn handle_click(
        &mut self,
        top_left: Point,
        click: Point,
        w: WindowPtr,
    ) -> Result<Option<???>>;
}
Enter fullscreen mode Exit fullscreen mode

The method needs to return whatever is coming out of these stored callbacks if the passed click falls inside this widget. Thing is, Callback<T> has gone and gotten itself all generic. This poses a problem, because we can't parameterize the trait itself with this type, like this:

pub trait Widget<T> {
    // ..
    /// Handle a click in this region
    fn handle_click(
        &mut self,
        top_left: Point,
        click: Point,
        w: WindowPtr,
    ) -> Result<Option<T>>;
}
Enter fullscreen mode Exit fullscreen mode

I need to be able to construct Widget trait objects, and that T is Sized. That's no good - a trait object is a dynamically-sized type and cannot have a known size at compile time. Depending on a monomorphized generic method means that you do have that information - you can't have your cake and eat it too. I kinda blew past this point last time but it bears a little more explanation.

Trait Objects

This library utilizes dynamic dispatch to allow for different applications and different backends to swap in and out using a common interface. To utilize it, you instantiate the following struct:

/// Top-level canvas engine object
pub struct WindowEngine {
    window: Rc<Box<dyn Window>>,
    element: Box<dyn Widget>,
}
Enter fullscreen mode Exit fullscreen mode

The Widget and Window traits just define some methods that need to be available - they don't describe any specific type. When we actually do put a real type in a Box to put in this struct, we completely lose the type information and only retain the trait information. A vtable is allocated instead with pointers to each method the trait specifies, and the pointer to it actually also contains this vtable pointer to complete the information needed to run your code. This means, though, that we can't use monomorphized generic types behind the pointer, because we literally don't even know what type we have. It's all handled through runtime reflection via these vtable pointers, you cannot use anything else. This is a good thing, it lets us define Widgets of all different shapes and sizes (memory-wise) and use them all identically. That's why Widget<T> is so problematic, though - it requires knowing all about what types are in play at compile time, which we emphatically do not.

Associated Types

Luckily there's a simple, ergonomic solution. Instead of parameterize the trait, you can just associate a type as part of the trait definition:

pub trait Widget {
    type MSG; // Associated message type

    /// Handle a click in this region
    fn handle_click(
        &mut self,
        top_left: Point,
        click: Point,
        w: WindowPtr,
    ) -> Result<Option<Self::MSG>>;
}
Enter fullscreen mode Exit fullscreen mode

Now we can parameterize the instantiated struct without needing the Widget definition itself to carry any baggage:

pub struct WindowEngine<T: 'static> {
    window: Rc<Box<dyn Window>>,
    element: Box<dyn Widget<MSG = T>>,
}
Enter fullscreen mode Exit fullscreen mode

Now when the vtable is created, it still has all the information it needs for your specific application's state management without constraining your type at all beyond 'static. The WindowEngine itself gets monomorphized with that type, i.e. WindowEngine<FiveDiceMessage>, so all Widgets this WindowEngine contains will use the same type.

When you write this type, you just fill it in at the application level:

impl Widget for Die {
    type MSG = FiveDiceMessage; // Specify associated type
    fn mount_widget(&self) -> MountedWidget<Self::MSG> {
         // ..
    }
    fn get_region(&self, top_left: Point, w: WindowPtr) -> WindowResult<Region> {
        // ..
    }
    fn handle_click(
        &mut self,
        top_left: Point,
        click: Point,
        w: WindowPtr,
    ) -> WindowResult<Option<Self::MSG>> {
        // ..
    }
}
Enter fullscreen mode Exit fullscreen mode

Associated types are a feature I knew about from using some standard library traits. For example, std::str::FromStr has you specify the error type to use:

impl FromStr for Color {
    type Err = WindowError; // Associated Err type

    fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
        match s {
            "black" => Ok(Color::new(0, 0, 0)),
            "red" => Ok(Color::new(255, 0, 0)),
            "blue" => Ok(Color::new(0, 0, 255)),
            "green" => Ok(Color::new(0, 255, 0)),
            // ..
        }
    }
}
Enter fullscreen mode Exit fullscreen mode

I hadn't thought about a use case where I'd need to write my own trait with one of these until I fell into the situation backwards. So it goes.

Lingering Concern - PhantomData?

There's one weird bit that I don't feel fully comfortable with. I have a generic Text widget designed to just plop a string on the canvas, that's not clickable. Its handle_click method doesn't return anything, so I use the None variant in the Widget impl:

impl<T: 'static> Widget for Text<T> {
    type MSG = T;
    // ..
    fn handle_click(&mut self, _: Point, _: Point, _: WindowPtr) -> Result<Option<Self::MSG>> {
        Ok(None)
    }
}
Enter fullscreen mode Exit fullscreen mode

However, this still requires that we parameterize Text with the message type of this particular widget tree, even though it's never used, because the return type still contains the Some(T) variant's type. This is what gets it to stop yelling at me:

/// A widget that just draws some text
pub struct Text<T> {
    phantom: std::marker::PhantomData<T>,
    text: String,
}

impl<T> Text<T> {
    pub fn new(s: &str) -> Self {
        Self {
            phantom: std::marker::PhantomData,
            text: s.into(),
        }
    }
}
Enter fullscreen mode Exit fullscreen mode

Per the docs, PhantomData is a zero-sized type that just "tells the compiler that your type acts as though it stores a value of type T, even though it doesn't really." This sounds like what I'm doing here, but I don't have a good sense of whether this is a kludge that I should try to refactor or the correct way to handle this situation.

Oh, my!

Questions and doubts aside, it all works as planned. Maybe, juuuust maybe, we'll hit up those procedural macros sometime.

Photo by 🇸🇮 Janko Ferlič - @specialdaddy on Unsplash

. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .