halley-wl 0.1.0

Wayland backend and rendering implementation for the Halley Wayland compositor.
use crate::compositor::interaction::ModState;
use halley_config::KeyModifiers;

/// Match an incoming XKB keycode against a stored evdev keycode.
/// Incoming codes from libinput/the backend are always XKB (evdev + 8).
/// Config and keybind tables store evdev codes, so we normalise here.
pub(crate) fn key_matches(actual: u32, evdev_code: u32) -> bool {
    evdev_code != 0 && actual == evdev_code + 8
}

/// Update modifier bookkeeping from a raw XKB keycode.
///
/// Each branch uses only the XKB code (evdev + 8). The old code had both the
/// evdev code and the XKB code in every branch (e.g. `29 || 37` for Left
/// Ctrl). That meant an ordinary key whose XKB code happened to equal some
/// modifier's *evdev* code would silently flip a modifier flag — most
/// visibly, XKB 54 (the letter C) was being treated as Right Shift's evdev
/// code 54, corrupting `right_shift_down` on every Ctrl+Shift+C press.
pub(crate) fn update_mod_state(mods: &mut ModState, code: u32, pressed: bool) {
    if code == 133 {
        // Left Super  (evdev 125 + 8)
        mods.left_super_down = pressed;
        mods.super_down = mods.left_super_down || mods.right_super_down;
    } else if code == 134 {
        // Right Super  (evdev 126 + 8)
        mods.right_super_down = pressed;
        mods.super_down = mods.left_super_down || mods.right_super_down;
    } else if code == 64 {
        // Left Alt  (evdev 56 + 8)
        mods.left_alt_down = pressed;
        mods.alt_down = mods.left_alt_down || mods.right_alt_down;
    } else if code == 108 {
        // Right Alt / AltGr  (evdev 100 + 8)
        mods.right_alt_down = pressed;
        mods.alt_down = mods.left_alt_down || mods.right_alt_down;
    } else if code == 37 {
        // Left Ctrl  (evdev 29 + 8)
        mods.left_ctrl_down = pressed;
        mods.ctrl_down = mods.left_ctrl_down || mods.right_ctrl_down;
    } else if code == 105 {
        // Right Ctrl  (evdev 97 + 8)
        mods.right_ctrl_down = pressed;
        mods.ctrl_down = mods.left_ctrl_down || mods.right_ctrl_down;
    } else if code == 50 {
        // Left Shift  (evdev 42 + 8)
        mods.left_shift_down = pressed;
        mods.shift_down = mods.left_shift_down || mods.right_shift_down;
    } else if code == 62 {
        // Right Shift  (evdev 54 + 8)
        mods.right_shift_down = pressed;
        mods.shift_down = mods.left_shift_down || mods.right_shift_down;
    }
}

pub(crate) fn modifier_active(mods: &ModState, need: KeyModifiers) -> bool {
    (!need.super_key || mods.super_down)
        && (!need.left_super || mods.left_super_down)
        && (!need.right_super || mods.right_super_down)
        && (!need.alt || mods.alt_down)
        && (!need.left_alt || mods.left_alt_down)
        && (!need.right_alt || mods.right_alt_down)
        && (!need.ctrl || mods.ctrl_down)
        && (!need.left_ctrl || mods.left_ctrl_down)
        && (!need.right_ctrl || mods.right_ctrl_down)
        && (!need.shift || mods.shift_down)
        && (!need.left_shift || mods.left_shift_down)
        && (!need.right_shift || mods.right_shift_down)
}

fn modifier_family_exact(
    actual_left: bool,
    actual_right: bool,
    need_any: bool,
    need_left: bool,
    need_right: bool,
) -> bool {
    if need_left && !actual_left {
        return false;
    }
    if need_right && !actual_right {
        return false;
    }
    if need_any && !(actual_left || actual_right) {
        return false;
    }

    let allow_left = need_any || need_left;
    let allow_right = need_any || need_right;

    (!actual_left || allow_left) && (!actual_right || allow_right)
}

pub(crate) fn modifier_exact(mods: &ModState, need: KeyModifiers) -> bool {
    modifier_family_exact(
        mods.left_super_down,
        mods.right_super_down,
        need.super_key,
        need.left_super,
        need.right_super,
    ) && modifier_family_exact(
        mods.left_alt_down,
        mods.right_alt_down,
        need.alt,
        need.left_alt,
        need.right_alt,
    ) && modifier_family_exact(
        mods.left_ctrl_down,
        mods.right_ctrl_down,
        need.ctrl,
        need.left_ctrl,
        need.right_ctrl,
    ) && modifier_family_exact(
        mods.left_shift_down,
        mods.right_shift_down,
        need.shift,
        need.left_shift,
        need.right_shift,
    )
}

#[cfg(test)]
mod tests {
    use super::modifier_exact;
    use crate::compositor::interaction::ModState;
    use halley_config::KeyModifiers;

    #[test]
    fn exact_modifier_matching_rejects_extra_shift() {
        let mods = ModState {
            left_super_down: true,
            super_down: true,
            left_shift_down: true,
            shift_down: true,
            ..ModState::default()
        };

        assert!(!modifier_exact(
            &mods,
            KeyModifiers {
                super_key: true,
                ..KeyModifiers::default()
            }
        ));
    }

    #[test]
    fn exact_modifier_matching_still_allows_generic_sided_match() {
        let mods = ModState {
            left_alt_down: true,
            alt_down: true,
            ..ModState::default()
        };

        assert!(modifier_exact(
            &mods,
            KeyModifiers {
                alt: true,
                ..KeyModifiers::default()
            }
        ));
    }
}

#[inline]
pub(crate) fn is_modifier_keycode(code: u32) -> bool {
    matches!(
        code,
        37 | 105 | 50 | 62 | 64 | 108 | 133 | 134 | 66 | 77 | 78
    )
}