sdl3 0.18.0

Bindings to SDL3, a cross-platform library to abstract the platform-specific details for building applications.
Documentation
fn main() -> Result<(), Box<dyn std::error::Error>> {
    let sdl_context = sdl3::init()?;
    let joystick_subsystem = sdl_context.joystick()?;
    let haptic_subsystem = sdl_context.haptic()?;

    let joysticks = joystick_subsystem
        .joysticks()
        .map_err(|e| format!("can't enumerate joysticks: {e}"))?;

    println!("{} joysticks available", joysticks.len());

    // Iterate over all available joysticks and stop once we manage to open one.
    let joystick_id = joysticks
        .into_iter()
        .find_map(|id| match joystick_subsystem.open(id) {
            Ok(c) => {
                println!("Success: opened \"{}\"", c.name());
                Some(id)
            }
            Err(e) => {
                println!("failed: {e:?}");
                None
            }
        })
        .expect("Couldn't open any joystick");

    let mut haptic = haptic_subsystem
        .open_from_joystick_id(joystick_id.into())
        .map_err(|e| e.to_string())?;

    for event in sdl_context.event_pump()?.wait_iter() {
        use sdl3::event::Event;

        match event {
            Event::JoyAxisMotion {
                axis_idx,
                value: val,
                ..
            } => {
                // Axis motion is an absolute value in the range
                // [-32768, 32767]. Let's simulate a very rough dead
                // zone to ignore spurious events.
                let dead_zone = 10_000;
                if val > dead_zone || val < -dead_zone {
                    println!("Axis {axis_idx} moved to {val}");
                }
            }
            Event::JoyButtonDown { button_idx, .. } => {
                println!("Button {button_idx} down");
                haptic.rumble_play(0.5, 500);
            }
            Event::JoyButtonUp { button_idx, .. } => println!("Button {button_idx} up"),
            Event::JoyHatMotion { hat_idx, state, .. } => {
                println!("Hat {hat_idx} moved to {state:?}")
            }
            Event::Quit { .. } => break,
            _ => (),
        }
    }

    Ok(())
}