1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
use std::ffi::c_int;
use std::sync::Arc;

use x11_dl::xlib::{self, XEvent, XGenericEventCookie};

use crate::platform_impl::x11::XConnection;

/// XEvents of type GenericEvent store their actual data in an XGenericEventCookie data structure.
/// This is a wrapper to extract the cookie from a GenericEvent XEvent and release the cookie data
/// once it has been processed
pub struct GenericEventCookie {
    cookie: XGenericEventCookie,
    xconn: Arc<XConnection>,
}

impl GenericEventCookie {
    pub fn from_event(xconn: Arc<XConnection>, event: XEvent) -> Option<GenericEventCookie> {
        unsafe {
            let mut cookie: XGenericEventCookie = From::from(event);
            if (xconn.xlib.XGetEventData)(xconn.display, &mut cookie) == xlib::True {
                Some(GenericEventCookie { cookie, xconn })
            } else {
                None
            }
        }
    }

    #[inline]
    pub fn extension(&self) -> u8 {
        self.cookie.extension as u8
    }

    #[inline]
    pub fn evtype(&self) -> c_int {
        self.cookie.evtype
    }

    /// Borrow inner event data as `&T`.
    ///
    /// ## SAFETY
    ///
    /// The caller must ensure that the event has the `T` inside of it.
    #[inline]
    pub unsafe fn as_event<T>(&self) -> &T {
        unsafe { &*(self.cookie.data as *const _) }
    }
}

impl Drop for GenericEventCookie {
    fn drop(&mut self) {
        unsafe {
            (self.xconn.xlib.XFreeEventData)(self.xconn.display, &mut self.cookie);
        }
    }
}