use std::io;
use std::string::{String, ToString};
use crate::{
Event, KeyCode, KeyEvent, KeyEventKind, KeyEventState, KeyModifiers, MediaKeyCode,
ModifierDirection, ModifierKeyCode, MouseButton, MouseEvent, MouseEventKind, ScrollDirection,
};
fn could_not_parse_event_error() -> io::Error {
io::Error::other("Could not parse event.")
}
impl Event {
pub fn parse_from(buffer: &[u8]) -> io::Result<Option<Self>> {
if buffer.is_empty() {
return Ok(None);
}
match buffer[0] {
b'\x1B' => {
if buffer.len() == 1 {
Ok(Some(Self::Key(KeyCode::Esc.into())))
} else {
match buffer[1] {
b'O' => {
if buffer.len() == 2 {
Ok(None)
} else {
match buffer[2] {
b'D' => Ok(Some(Self::Key(KeyCode::Left.into()))),
b'C' => Ok(Some(Self::Key(KeyCode::Right.into()))),
b'A' => Ok(Some(Self::Key(KeyCode::Up.into()))),
b'B' => Ok(Some(Self::Key(KeyCode::Down.into()))),
b'H' => Ok(Some(Self::Key(KeyCode::Home.into()))),
b'F' => Ok(Some(Self::Key(KeyCode::End.into()))),
val @ b'P'..=b'S' => {
Ok(Some(Self::Key(KeyCode::F(1 + val - b'P').into())))
}
_ => Err(could_not_parse_event_error()),
}
}
}
b'[' => parse_csi(buffer),
b'\x1B' => {
if buffer.len() == 2 {
Ok(Some(Self::Key(
KeyEvent::new(KeyCode::Esc).modifiers(KeyModifiers::ALT),
)))
} else {
match &buffer[2..] {
b"[Z" => Ok(Some(Self::Key(
KeyEvent::new(KeyCode::Tab)
.modifiers(KeyModifiers::SHIFT | KeyModifiers::ALT),
))),
_ => Err(could_not_parse_event_error()),
}
}
}
_ => Self::parse_from(&buffer[1..]).map(|event_option| {
event_option.map(|event| {
if let Self::Key(key_event) = event {
let mut alt_key_event = key_event;
alt_key_event.modifiers |= KeyModifiers::ALT;
Self::Key(alt_key_event)
} else {
event
}
})
}),
}
}
}
b'\r' => Ok(Some(Self::Key(KeyCode::Enter.into()))),
b'\t' => Ok(Some(Self::Key(KeyCode::Tab.into()))),
b'\x7F' => Ok(Some(Self::Key(KeyCode::Backspace.into()))),
c @ b'\x01'..=b'\x1A' => Ok(Some(Self::Key(
KeyEvent::new(KeyCode::Char((c - 0x1 + b'a') as char))
.modifiers(KeyModifiers::CTRL),
))),
c @ b'\x1C'..=b'\x1F' => Ok(Some(Self::Key(
KeyEvent::new(KeyCode::Char((c - 0x1C + b'4') as char))
.modifiers(KeyModifiers::CTRL),
))),
b'\0' => Ok(Some(Self::Key(
KeyEvent::new(KeyCode::Char(' ')).modifiers(KeyModifiers::CTRL),
))),
_ => parse_utf8_char(buffer).map(|maybe_char| {
maybe_char
.map(KeyCode::Char)
.map(char_code_to_event)
.map(Event::Key)
}),
}
}
}
fn char_code_to_event(code: KeyCode) -> KeyEvent {
let modifiers = match code {
KeyCode::Char(c) if c.is_uppercase() => KeyModifiers::SHIFT,
_ => KeyModifiers::empty(),
};
KeyEvent::new(code).modifiers(modifiers)
}
pub(crate) fn parse_csi(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B["));
if buffer.len() == 2 {
return Ok(None);
}
let input_event = match buffer[2] {
b'[' => {
if buffer.len() == 3 {
None
} else {
match buffer[3] {
val @ b'A'..=b'E' => Some(Event::Key(KeyCode::F(1 + val - b'A').into())),
_ => return Err(could_not_parse_event_error()),
}
}
}
b'D' => Some(Event::Key(KeyCode::Left.into())),
b'C' => Some(Event::Key(KeyCode::Right.into())),
b'A' => Some(Event::Key(KeyCode::Up.into())),
b'B' => Some(Event::Key(KeyCode::Down.into())),
b'H' => Some(Event::Key(KeyCode::Home.into())),
b'F' => Some(Event::Key(KeyCode::End.into())),
b'Z' => Some(Event::Key(
KeyEvent::new(KeyCode::Tab).modifiers(KeyModifiers::SHIFT),
)),
b'M' => return parse_csi_normal_mouse(buffer),
b'<' => return parse_csi_sgr_mouse(buffer),
b'I' => Some(Event::FocusGained),
b'O' => Some(Event::FocusLost),
b';' => return parse_csi_modifier_key_code(buffer),
b'P' => Some(Event::Key(KeyCode::F(1).into())),
b'Q' => Some(Event::Key(KeyCode::F(2).into())),
b'R' => Some(Event::Key(KeyCode::F(3).into())),
b'S' => Some(Event::Key(KeyCode::F(4).into())),
b'?' => match buffer[buffer.len() - 1] {
b'u' => return Err(could_not_parse_event_error()),
b'c' => return Err(could_not_parse_event_error()),
_ => None,
},
b'0'..=b'9' => {
if buffer.len() == 3 {
None
} else {
let last_byte = buffer[buffer.len() - 1];
if !(64..=126).contains(&last_byte) {
None
} else {
if buffer.starts_with(b"\x1B[200~") {
return parse_csi_bracketed_paste(buffer);
}
match last_byte {
b'M' => return parse_csi_rxvt_mouse(buffer),
b'~' => return parse_csi_special_key_code(buffer),
b'u' => return parse_csi_u_encoded_key_code(buffer),
_ => return parse_csi_modifier_key_code(buffer),
}
}
}
}
_ => return Err(could_not_parse_event_error()),
};
Ok(input_event)
}
pub(crate) fn next_parsed<T>(iter: &mut dyn Iterator<Item = &str>) -> io::Result<T>
where
T: std::str::FromStr,
{
iter.next()
.ok_or_else(could_not_parse_event_error)?
.parse::<T>()
.map_err(|_| could_not_parse_event_error())
}
fn modifier_and_kind_parsed(iter: &mut dyn Iterator<Item = &str>) -> io::Result<(u8, u8)> {
let mut sub_split = iter
.next()
.ok_or_else(could_not_parse_event_error)?
.split(':');
let modifier_mask = next_parsed::<u8>(&mut sub_split)?;
if let Ok(kind_code) = next_parsed::<u8>(&mut sub_split) {
Ok((modifier_mask, kind_code))
} else {
Ok((modifier_mask, 1))
}
}
fn parse_modifiers(mask: u8) -> KeyModifiers {
let modifier_mask = mask.saturating_sub(1);
let mut modifiers = KeyModifiers::empty();
if modifier_mask & 1 != 0 {
modifiers |= KeyModifiers::SHIFT;
}
if modifier_mask & 2 != 0 {
modifiers |= KeyModifiers::ALT;
}
if modifier_mask & 4 != 0 {
modifiers |= KeyModifiers::CTRL;
}
if modifier_mask & 8 != 0 {
modifiers |= KeyModifiers::SUPER;
}
if modifier_mask & 16 != 0 {
modifiers |= KeyModifiers::HYPER;
}
if modifier_mask & 32 != 0 {
modifiers |= KeyModifiers::META;
}
modifiers
}
fn parse_modifiers_to_state(mask: u8) -> KeyEventState {
let modifier_mask = mask.saturating_sub(1);
let mut state = KeyEventState::empty();
if modifier_mask & 64 != 0 {
state |= KeyEventState::CAPS_LOCK;
}
if modifier_mask & 128 != 0 {
state |= KeyEventState::NUM_LOCK;
}
state
}
fn parse_key_event_kind(kind: u8) -> KeyEventKind {
match kind {
1 => KeyEventKind::Press,
2 => KeyEventKind::Repeat,
3 => KeyEventKind::Release,
_ => KeyEventKind::Press,
}
}
pub(crate) fn parse_csi_modifier_key_code(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B["));
let s = std::str::from_utf8(&buffer[2..buffer.len() - 1])
.map_err(|_| could_not_parse_event_error())?;
let mut split = s.split(';');
split.next();
let (modifiers, kind) =
if let Ok((modifier_mask, kind_code)) = modifier_and_kind_parsed(&mut split) {
(
parse_modifiers(modifier_mask),
parse_key_event_kind(kind_code),
)
} else if buffer.len() > 3 {
(
parse_modifiers(
(buffer[buffer.len() - 2] as char)
.to_digit(10)
.ok_or_else(could_not_parse_event_error)? as u8,
),
KeyEventKind::Press,
)
} else {
(KeyModifiers::NONE, KeyEventKind::Press)
};
let key = buffer[buffer.len() - 1];
let keycode = match key {
b'A' => KeyCode::Up,
b'B' => KeyCode::Down,
b'C' => KeyCode::Right,
b'D' => KeyCode::Left,
b'F' => KeyCode::End,
b'H' => KeyCode::Home,
b'P' => KeyCode::F(1),
b'Q' => KeyCode::F(2),
b'R' => KeyCode::F(3),
b'S' => KeyCode::F(4),
_ => return Err(could_not_parse_event_error()),
};
let input_event = Event::Key(KeyEvent::new(keycode).modifiers(modifiers).kind(kind));
Ok(Some(input_event))
}
fn translate_functional_key_code(codepoint: u32) -> Option<(KeyCode, KeyEventState)> {
if let Some(keycode) = match codepoint {
57399 => Some(KeyCode::Char('0')),
57400 => Some(KeyCode::Char('1')),
57401 => Some(KeyCode::Char('2')),
57402 => Some(KeyCode::Char('3')),
57403 => Some(KeyCode::Char('4')),
57404 => Some(KeyCode::Char('5')),
57405 => Some(KeyCode::Char('6')),
57406 => Some(KeyCode::Char('7')),
57407 => Some(KeyCode::Char('8')),
57408 => Some(KeyCode::Char('9')),
57409 => Some(KeyCode::Char('.')),
57410 => Some(KeyCode::Char('/')),
57411 => Some(KeyCode::Char('*')),
57412 => Some(KeyCode::Char('-')),
57413 => Some(KeyCode::Char('+')),
57414 => Some(KeyCode::Enter),
57415 => Some(KeyCode::Char('=')),
57416 => Some(KeyCode::Char(',')),
57417 => Some(KeyCode::Left),
57418 => Some(KeyCode::Right),
57419 => Some(KeyCode::Up),
57420 => Some(KeyCode::Down),
57421 => Some(KeyCode::PageUp),
57422 => Some(KeyCode::PageDown),
57423 => Some(KeyCode::Home),
57424 => Some(KeyCode::End),
57425 => Some(KeyCode::Insert),
57426 => Some(KeyCode::Delete),
57427 => Some(KeyCode::KeypadBegin),
_ => None,
} {
return Some((keycode, KeyEventState::KEYPAD));
}
if let Some(keycode) = match codepoint {
57358 => Some(KeyCode::CapsLock),
57359 => Some(KeyCode::ScrollLock),
57360 => Some(KeyCode::NumLock),
57361 => Some(KeyCode::PrintScreen),
57362 => Some(KeyCode::Pause),
57363 => Some(KeyCode::Menu),
57376 => Some(KeyCode::F(13)),
57377 => Some(KeyCode::F(14)),
57378 => Some(KeyCode::F(15)),
57379 => Some(KeyCode::F(16)),
57380 => Some(KeyCode::F(17)),
57381 => Some(KeyCode::F(18)),
57382 => Some(KeyCode::F(19)),
57383 => Some(KeyCode::F(20)),
57384 => Some(KeyCode::F(21)),
57385 => Some(KeyCode::F(22)),
57386 => Some(KeyCode::F(23)),
57387 => Some(KeyCode::F(24)),
57388 => Some(KeyCode::F(25)),
57389 => Some(KeyCode::F(26)),
57390 => Some(KeyCode::F(27)),
57391 => Some(KeyCode::F(28)),
57392 => Some(KeyCode::F(29)),
57393 => Some(KeyCode::F(30)),
57394 => Some(KeyCode::F(31)),
57395 => Some(KeyCode::F(32)),
57396 => Some(KeyCode::F(33)),
57397 => Some(KeyCode::F(34)),
57398 => Some(KeyCode::F(35)),
57428 => Some(KeyCode::Media(MediaKeyCode::Play)),
57429 => Some(KeyCode::Media(MediaKeyCode::Pause)),
57430 => Some(KeyCode::Media(MediaKeyCode::PlayPause)),
57431 => Some(KeyCode::Media(MediaKeyCode::Reverse)),
57432 => Some(KeyCode::Media(MediaKeyCode::Stop)),
57433 => Some(KeyCode::Media(MediaKeyCode::FastForward)),
57434 => Some(KeyCode::Media(MediaKeyCode::Rewind)),
57435 => Some(KeyCode::Media(MediaKeyCode::TrackNext)),
57436 => Some(KeyCode::Media(MediaKeyCode::TrackPrevious)),
57437 => Some(KeyCode::Media(MediaKeyCode::Record)),
57438 => Some(KeyCode::Media(MediaKeyCode::LowerVolume)),
57439 => Some(KeyCode::Media(MediaKeyCode::RaiseVolume)),
57440 => Some(KeyCode::Media(MediaKeyCode::MuteVolume)),
57441 => Some(KeyCode::Modifier(
ModifierKeyCode::Shift,
ModifierDirection::Left,
)),
57442 => Some(KeyCode::Modifier(
ModifierKeyCode::Control,
ModifierDirection::Left,
)),
57443 => Some(KeyCode::Modifier(
ModifierKeyCode::Alt,
ModifierDirection::Left,
)),
57444 => Some(KeyCode::Modifier(
ModifierKeyCode::Super,
ModifierDirection::Left,
)),
57445 => Some(KeyCode::Modifier(
ModifierKeyCode::Hyper,
ModifierDirection::Left,
)),
57446 => Some(KeyCode::Modifier(
ModifierKeyCode::Meta,
ModifierDirection::Left,
)),
57447 => Some(KeyCode::Modifier(
ModifierKeyCode::Shift,
ModifierDirection::Right,
)),
57448 => Some(KeyCode::Modifier(
ModifierKeyCode::Control,
ModifierDirection::Right,
)),
57449 => Some(KeyCode::Modifier(
ModifierKeyCode::Alt,
ModifierDirection::Right,
)),
57450 => Some(KeyCode::Modifier(
ModifierKeyCode::Super,
ModifierDirection::Right,
)),
57451 => Some(KeyCode::Modifier(
ModifierKeyCode::Hyper,
ModifierDirection::Right,
)),
57452 => Some(KeyCode::Modifier(
ModifierKeyCode::Meta,
ModifierDirection::Right,
)),
57453 => Some(KeyCode::Modifier(
ModifierKeyCode::IsoLevel3Shift,
ModifierDirection::Unknown,
)),
57454 => Some(KeyCode::Modifier(
ModifierKeyCode::IsoLevel5Shift,
ModifierDirection::Unknown,
)),
_ => None,
} {
return Some((keycode, KeyEventState::empty()));
}
None
}
pub(crate) fn parse_csi_u_encoded_key_code(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B[")); assert!(buffer.ends_with(b"u"));
let s = std::str::from_utf8(&buffer[2..buffer.len() - 1])
.map_err(|_| could_not_parse_event_error())?;
let mut split = s.split(';');
let mut codepoints = split
.next()
.ok_or_else(could_not_parse_event_error)?
.split(':');
let codepoint = codepoints
.next()
.ok_or_else(could_not_parse_event_error)?
.parse::<u32>()
.map_err(|_| could_not_parse_event_error())?;
let (mut modifiers, kind, state_from_modifiers) =
if let Ok((modifier_mask, kind_code)) = modifier_and_kind_parsed(&mut split) {
(
parse_modifiers(modifier_mask),
parse_key_event_kind(kind_code),
parse_modifiers_to_state(modifier_mask),
)
} else {
(KeyModifiers::NONE, KeyEventKind::Press, KeyEventState::NONE)
};
let (mut keycode, state_from_keycode) = {
if let Some((special_key_code, state)) = translate_functional_key_code(codepoint) {
(special_key_code, state)
} else if let Some(c) = char::from_u32(codepoint) {
(
match c {
'\x1B' => KeyCode::Esc,
'\r' => KeyCode::Enter,
'\t' => KeyCode::Tab,
'\x7F' => KeyCode::Backspace,
_ => KeyCode::Char(c),
},
KeyEventState::empty(),
)
} else {
return Err(could_not_parse_event_error());
}
};
if let KeyCode::Modifier(modifier_keycode, _) = keycode {
match modifier_keycode {
ModifierKeyCode::Alt => {
modifiers.set(KeyModifiers::ALT, true);
}
ModifierKeyCode::Control => {
modifiers.set(KeyModifiers::CTRL, true);
}
ModifierKeyCode::Shift => {
modifiers.set(KeyModifiers::SHIFT, true);
}
ModifierKeyCode::Super => {
modifiers.set(KeyModifiers::SUPER, true);
}
ModifierKeyCode::Hyper => {
modifiers.set(KeyModifiers::HYPER, true);
}
ModifierKeyCode::Meta => {
modifiers.set(KeyModifiers::META, true);
}
_ => {}
}
}
if modifiers.contains(KeyModifiers::SHIFT)
&& let Some(shifted_c) = codepoints
.next()
.and_then(|codepoint| codepoint.parse::<u32>().ok())
.and_then(char::from_u32)
{
keycode = KeyCode::Char(shifted_c);
modifiers.set(KeyModifiers::SHIFT, false);
}
let input_event = Event::Key(
KeyEvent::new(keycode)
.modifiers(modifiers)
.kind(kind)
.state(state_from_keycode | state_from_modifiers),
);
Ok(Some(input_event))
}
pub(crate) fn parse_csi_special_key_code(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B[")); assert!(buffer.ends_with(b"~"));
let s = std::str::from_utf8(&buffer[2..buffer.len() - 1])
.map_err(|_| could_not_parse_event_error())?;
let mut split = s.split(';');
let first = next_parsed::<u8>(&mut split)?;
let (modifiers, kind, state) =
if let Ok((modifier_mask, kind_code)) = modifier_and_kind_parsed(&mut split) {
(
parse_modifiers(modifier_mask),
parse_key_event_kind(kind_code),
parse_modifiers_to_state(modifier_mask),
)
} else {
(KeyModifiers::NONE, KeyEventKind::Press, KeyEventState::NONE)
};
let keycode = match first {
1 | 7 => KeyCode::Home,
2 => KeyCode::Insert,
3 => KeyCode::Delete,
4 | 8 => KeyCode::End,
5 => KeyCode::PageUp,
6 => KeyCode::PageDown,
v @ 11..=15 => KeyCode::F(v - 10),
v @ 17..=21 => KeyCode::F(v - 11),
v @ 23..=26 => KeyCode::F(v - 12),
v @ 28..=29 => KeyCode::F(v - 15),
v @ 31..=34 => KeyCode::F(v - 17),
_ => return Err(could_not_parse_event_error()),
};
let input_event = Event::Key(
KeyEvent::new(keycode)
.modifiers(modifiers)
.kind(kind)
.state(state),
);
Ok(Some(input_event))
}
pub(crate) fn parse_csi_rxvt_mouse(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B[")); assert!(buffer.ends_with(b"M"));
let s = std::str::from_utf8(&buffer[2..buffer.len() - 1])
.map_err(|_| could_not_parse_event_error())?;
let mut split = s.split(';');
let cb = next_parsed::<u8>(&mut split)?
.checked_sub(32)
.ok_or_else(could_not_parse_event_error)?;
let (kind, modifiers) = parse_cb(cb)?;
let cx = next_parsed::<u16>(&mut split)? - 1;
let cy = next_parsed::<u16>(&mut split)? - 1;
Ok(Some(Event::Mouse(MouseEvent {
kind,
column: cx,
row: cy,
modifiers,
})))
}
pub(crate) fn parse_csi_normal_mouse(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B[M"));
if buffer.len() < 6 {
return Ok(None);
}
let cb = buffer[3]
.checked_sub(32)
.ok_or_else(could_not_parse_event_error)?;
let (kind, modifiers) = parse_cb(cb)?;
let cx = u16::from(buffer[4].saturating_sub(33));
let cy = u16::from(buffer[5].saturating_sub(33));
Ok(Some(Event::Mouse(MouseEvent {
kind,
column: cx,
row: cy,
modifiers,
})))
}
pub(crate) fn parse_csi_sgr_mouse(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B[<"));
if !buffer.ends_with(b"m") && !buffer.ends_with(b"M") {
return Ok(None);
}
let s = std::str::from_utf8(&buffer[3..buffer.len() - 1])
.map_err(|_| could_not_parse_event_error())?;
let mut split = s.split(';');
let cb = next_parsed::<u8>(&mut split)?;
let (kind, modifiers) = parse_cb(cb)?;
let cx = next_parsed::<u16>(&mut split)? - 1;
let cy = next_parsed::<u16>(&mut split)? - 1;
let kind = if buffer.last() == Some(&b'm') {
match kind {
MouseEventKind::Down(button) => MouseEventKind::Up(button),
other => other,
}
} else {
kind
};
Ok(Some(Event::Mouse(MouseEvent {
kind,
column: cx,
row: cy,
modifiers,
})))
}
fn parse_cb(cb: u8) -> io::Result<(MouseEventKind, KeyModifiers)> {
let button_number = (cb & 0b0000_0011) | ((cb & 0b1100_0000) >> 4);
let dragging = cb & 0b0010_0000 == 0b0010_0000;
let kind = match (button_number, dragging) {
(0, false) => MouseEventKind::Down(MouseButton::Left),
(1, false) => MouseEventKind::Down(MouseButton::Middle),
(2, false) => MouseEventKind::Down(MouseButton::Right),
(0, true) => MouseEventKind::Drag(MouseButton::Left),
(1, true) => MouseEventKind::Drag(MouseButton::Middle),
(2, true) => MouseEventKind::Drag(MouseButton::Right),
(3, false) => MouseEventKind::Up(MouseButton::Left),
(3, true) | (4, true) | (5, true) => MouseEventKind::Moved,
(4, false) => MouseEventKind::Scroll(ScrollDirection::Up),
(5, false) => MouseEventKind::Scroll(ScrollDirection::Down),
(6, false) => MouseEventKind::Scroll(ScrollDirection::Left),
(7, false) => MouseEventKind::Scroll(ScrollDirection::Right),
_ => return Err(could_not_parse_event_error()),
};
let mut modifiers = KeyModifiers::empty();
if cb & 0b0000_0100 == 0b0000_0100 {
modifiers |= KeyModifiers::SHIFT;
}
if cb & 0b0000_1000 == 0b0000_1000 {
modifiers |= KeyModifiers::ALT;
}
if cb & 0b0001_0000 == 0b0001_0000 {
modifiers |= KeyModifiers::CTRL;
}
Ok((kind, modifiers))
}
pub(crate) fn parse_csi_bracketed_paste(buffer: &[u8]) -> io::Result<Option<Event>> {
assert!(buffer.starts_with(b"\x1B[200~"));
if !buffer.ends_with(b"\x1b[201~") {
Ok(None)
} else {
let paste = String::from_utf8_lossy(&buffer[6..buffer.len() - 6]).to_string();
Ok(Some(Event::Paste(paste)))
}
}
pub(crate) fn parse_utf8_char(buffer: &[u8]) -> io::Result<Option<char>> {
match std::str::from_utf8(buffer) {
Ok(s) => {
let ch = s.chars().next().ok_or_else(could_not_parse_event_error)?;
Ok(Some(ch))
}
Err(_) => {
let required_bytes = match buffer[0] {
(0x00..=0x7F) => 1, (0xC0..=0xDF) => 2, (0xE0..=0xEF) => 3, (0xF0..=0xF7) => 4, (0x80..=0xBF) | (0xF8..=0xFF) => return Err(could_not_parse_event_error()),
};
if required_bytes > 1 && buffer.len() > 1 {
for byte in &buffer[1..] {
if byte & !0b0011_1111 != 0b1000_0000 {
return Err(could_not_parse_event_error());
}
}
}
if buffer.len() < required_bytes {
Ok(None)
} else {
Err(could_not_parse_event_error())
}
}
}
}