#include "Ecore.h" #include #include #include /* private funtion prototypes */ static void ecore_event_x_handle_events(int fd); static void ecore_event_x_translate_events(XEvent * events, int num_events); static void ecore_event_key_down_free(void *event); static void ecore_event_key_up_free(void *event); static void ecore_event_generic_free(void *event); static void ecore_event_dnd_drop_request_free(void *event); static void ecore_event_paste_request_free(void *event); static void ecore_event_x_handle_keypress(XEvent * xevent); static void ecore_event_x_handle_keyrelease(XEvent * xevent); static void ecore_event_x_handle_button_press(XEvent * xevent); static void ecore_event_x_handle_button_release(XEvent * xevent); static void ecore_event_x_handle_motion_notify(XEvent * xevent); static void ecore_event_x_handle_enter_notify(XEvent * xevent); static void ecore_event_x_handle_leave_notify(XEvent * xevent); static void ecore_event_x_handle_focus_in(XEvent * xevent); static void ecore_event_x_handle_focus_out(XEvent * xevent); static void ecore_event_x_handle_expose(XEvent * xevent); static void ecore_event_x_handle_visibility_notify(XEvent * xevent); static void ecore_event_x_handle_create_notify(XEvent * xevent); static void ecore_event_x_handle_destroy_notify(XEvent * xevent); static void ecore_event_x_handle_unmap_notify(XEvent * xevent); static void ecore_event_x_handle_map_notify(XEvent * xevent); static void ecore_event_x_handle_map_request(XEvent * xevent); static void ecore_event_x_handle_reparent_notify(XEvent * xevent); static void ecore_event_x_handle_configure_notify(XEvent * xevent); static void ecore_event_x_handle_configure_request(XEvent * xevent); static void ecore_event_x_handle_circulate_notify(XEvent * xevent); static void ecore_event_x_handle_circulate_request(XEvent * xevent); static void ecore_event_x_handle_property_notify(XEvent * xevent); static void ecore_event_x_handle_colormap_notify(XEvent * xevent); static void ecore_event_x_handle_selection_notify(XEvent * xevent); static void ecore_event_x_handle_selection_clear(XEvent * xevent); static void ecore_event_x_handle_selection_request(XEvent * xevent); static void ecore_event_x_handle_client_message(XEvent * xevent); static void ecore_event_x_handle_shape_change(XEvent * xevent); static int max_event_id = 0; static void (**event_translator) (XEvent * event) = NULL; static int lock_mask_scroll = 0, lock_mask_num = 0, lock_mask_caps = 0; static int mod_mask_shift = 0, mod_mask_ctrl = 0, mod_mask_alt = 0, mod_mask_win = 0; /* convenience macros */ #define GETSET_MODS(state, mods) \ (mods) = ECORE_EVENT_KEY_MODIFIER_NONE;\ if ((state) & mod_mask_shift) { ecore_mod_shift_set(1); (mods) |= ECORE_EVENT_KEY_MODIFIER_SHIFT; }\ else ecore_mod_shift_set(0);\ if ((state) & mod_mask_ctrl) { ecore_mod_ctrl_set(1); (mods) |= ECORE_EVENT_KEY_MODIFIER_CTRL; }\ else ecore_mod_ctrl_set(0);\ if ((state) & mod_mask_alt) { ecore_mod_alt_set(1); (mods) |= ECORE_EVENT_KEY_MODIFIER_ALT; }\ else ecore_mod_alt_set(0);\ if ((state) & mod_mask_win) { ecore_mod_win_set(1); (mods) |= ECORE_EVENT_KEY_MODIFIER_WIN; }\ else ecore_mod_win_set(0); /* public functions */ /* initialise event handling for the fd X is on */ void ecore_event_x_init(void) { int i, shape_event_id, current_lock; shape_event_id = max_event_id = ecore_event_shape_get_id(); if (shape_event_id < LASTEvent) { max_event_id = LASTEvent; fprintf(stderr, "ERROR: No shape extension! This is BAD!\n"); } event_translator = NEW_PTR(max_event_id + 1); for (i = 0; i < max_event_id + 1; i++) event_translator[i] = NULL; event_translator[KeyPress] = ecore_event_x_handle_keypress; event_translator[KeyRelease] = ecore_event_x_handle_keyrelease; event_translator[ButtonPress] = ecore_event_x_handle_button_press; event_translator[ButtonRelease] = ecore_event_x_handle_button_release; event_translator[MotionNotify] = ecore_event_x_handle_motion_notify; event_translator[EnterNotify] = ecore_event_x_handle_enter_notify; event_translator[LeaveNotify] = ecore_event_x_handle_leave_notify; event_translator[FocusIn] = ecore_event_x_handle_focus_in; event_translator[FocusOut] = ecore_event_x_handle_focus_out; event_translator[Expose] = ecore_event_x_handle_expose; event_translator[VisibilityNotify] = ecore_event_x_handle_visibility_notify; event_translator[CreateNotify] = ecore_event_x_handle_create_notify; event_translator[DestroyNotify] = ecore_event_x_handle_destroy_notify; event_translator[UnmapNotify] = ecore_event_x_handle_unmap_notify; event_translator[MapNotify] = ecore_event_x_handle_map_notify; event_translator[MapRequest] = ecore_event_x_handle_map_request; event_translator[ReparentNotify] = ecore_event_x_handle_reparent_notify; event_translator[ConfigureNotify] = ecore_event_x_handle_configure_notify; event_translator[ConfigureRequest] = ecore_event_x_handle_configure_request; event_translator[CirculateNotify] = ecore_event_x_handle_circulate_notify; event_translator[CirculateRequest] = ecore_event_x_handle_circulate_request; event_translator[PropertyNotify] = ecore_event_x_handle_property_notify; event_translator[ColormapNotify] = ecore_event_x_handle_colormap_notify; event_translator[ClientMessage] = ecore_event_x_handle_client_message; event_translator[SelectionNotify] = ecore_event_x_handle_selection_notify; event_translator[SelectionClear] = ecore_event_x_handle_selection_clear; event_translator[SelectionRequest] = ecore_event_x_handle_selection_request; if (shape_event_id > SelectionRequest) event_translator[shape_event_id] = ecore_event_x_handle_shape_change; lock_mask_scroll = ecore_lock_mask_scroll_get(); lock_mask_num = ecore_lock_mask_num_get(); lock_mask_caps = ecore_lock_mask_caps_get(); mod_mask_shift = ecore_mod_mask_shift_get(); mod_mask_ctrl = ecore_mod_mask_ctrl_get(); mod_mask_alt = ecore_mod_mask_alt_get(); mod_mask_win = ecore_mod_mask_win_get(); /* HRRRMMM lets not do this ecorecore_keygrab("Num_Lock", ECORE_EVENT_KEY_MODIFIER_NONE, 1, 1); ecorecore_keygrab("Scroll_Lock", ECORE_EVENT_KEY_MODIFIER_NONE, 1, 1); ecorecore_keygrab("Caps_Lock", ECORE_EVENT_KEY_MODIFIER_NONE, 1, 1); */ current_lock = ecore_modifier_mask_get(); if (current_lock & lock_mask_scroll) ecore_lock_scroll_set(1); if (current_lock & lock_mask_num) ecore_lock_num_set(1); if (current_lock & lock_mask_caps) ecore_lock_caps_set(1); if (current_lock & mod_mask_shift) ecore_mod_shift_set(1); if (current_lock & mod_mask_ctrl) ecore_mod_ctrl_set(1); if (current_lock & mod_mask_alt) ecore_mod_alt_set(1); if (current_lock & mod_mask_win) ecore_mod_win_set(1); ecore_add_event_fd(ecore_x_get_fd(), ecore_event_x_handle_events); } /* private functions */ /* get all events onthe event queue and translate them */ static void ecore_event_x_handle_events(int fd) { int num_events = 0, size_events = 0; XEvent *events = NULL; /* while there are events in the queue */ while (ecore_events_pending()) { /* incriment our event count */ num_events++; /* if the numebr fo events is > than our buffer size then */ if (num_events > size_events) { /* increase the buffer size by 64 events */ size_events += 64; if (events) { REALLOC(events, XEvent, size_events)} else events = NEW(XEvent, size_events); } /* get the next event into the event buffer */ ecore_get_next_event(&(events[num_events - 1])); } /* call the XEvent -> Eevent translator */ if (events) { ecore_event_x_translate_events(events, num_events); /* if theres an event buffer - free it */ FREE(events); } return; fd = 0; } /* take an array of events and translate them into E events */ static void ecore_event_x_translate_events(XEvent * events, int num_events) { int i; for (i = 0; i < num_events; i++) { if ((events[i].type <= max_event_id) && (event_translator[events[i].type])) (*(event_translator[events[i].type])) (&(events[i])); } } static void ecore_event_key_down_free(void *event) { Ecore_Event_Key_Down *e; e = (Ecore_Event_Key_Down *) event; IF_FREE(e->key); IF_FREE(e->compose); FREE(e); } static void ecore_event_key_up_free(void *event) { Ecore_Event_Key_Up *e; e = (Ecore_Event_Key_Up *) event; IF_FREE(e->key); IF_FREE(e->compose); FREE(e); } static void ecore_event_generic_free(void *event) { FREE(event); } static void ecore_event_dnd_drop_request_free(void *event) { Ecore_Event_Dnd_Drop_Request *e; e = (Ecore_Event_Dnd_Drop_Request *) event; if (e->files) { int i; for (i = 0; i < e->num_files; i++) FREE(e->files[i]); } FREE(event); } static void ecore_event_paste_request_free(void *event) { Ecore_Event_Paste_Request *e; e = (Ecore_Event_Paste_Request *) event; IF_FREE(e->string); FREE(event); } static void ecore_event_x_handle_keypress(XEvent * xevent) { Ecore_Event_Key_Down *e; static KeyCode previous_code = 0; static Time previous_time = 0; /* avoid doubling events up from passive grabs */ if ((xevent->xkey.keycode == previous_code) && xevent->xkey.time == previous_time) return; previous_code = xevent->xkey.keycode; previous_time = xevent->xkey.time; /* if (ecore_keyget_keysym_from_keycode(xevent->xkey.keycode) == XK_Scroll_Lock) { if (ecore_lock_scroll_get()) ecore_lock_scroll_set(0); else ecore_lock_scroll_set(1); e_event_allow(ReplayKeyboard, xevent->xkey.time); ecore_flush(); } else if (ecore_keyget_keysym_from_keycode(xevent->xkey.keycode) == XK_Num_Lock) { if (ecore_lock_num_get()) ecore_lock_num_set(0); else ecore_lock_num_set(1); e_event_allow(ReplayKeyboard, xevent->xkey.time); ecore_flush(); } else if (ecore_keyget_keysym_from_keycode(xevent->xkey.keycode) == XK_Caps_Lock) { if (ecore_lock_caps_get()) ecore_lock_caps_set(0); else ecore_lock_caps_set(1); e_event_allow(ReplayKeyboard, xevent->xkey.time); ecore_flush(); } */ e = NEW(Ecore_Event_Key_Down, 1); e->win = xevent->xkey.window; e->root = xevent->xkey.root; GETSET_MODS(xevent->xkey.state, e->mods); e->time = xevent->xkey.time; e->key = ecore_key_get_string_from_keycode(xevent->xkey.keycode); { int val; char buf[256]; KeySym sym; XComposeStatus stat; val = XLookupString((XKeyEvent *) xevent, buf, sizeof(buf), &sym, &stat); if (val > 0) { buf[val] = 0; e->compose = strdup(buf); } else e->compose = NULL; } ecore_add_event(ECORE_EVENT_KEY_DOWN, e, ecore_event_key_down_free); } static void ecore_event_x_handle_keyrelease(XEvent * xevent) { Ecore_Event_Key_Up *e; static KeyCode previous_code = 0; static Time previous_time = 0; /* avoid doubling events up from passive grabs */ if ((xevent->xkey.keycode == previous_code) && xevent->xkey.time == previous_time) return; previous_code = xevent->xkey.keycode; previous_time = xevent->xkey.time; e = NEW(Ecore_Event_Key_Up, 1); e->win = xevent->xkey.window; e->root = xevent->xkey.root; GETSET_MODS(xevent->xkey.state, e->mods); e->time = xevent->xkey.time; e->key = ecore_key_get_string_from_keycode(xevent->xkey.keycode); { int val; char buf[256]; KeySym sym; XComposeStatus stat; val = XLookupString((XKeyEvent *) xevent, buf, sizeof(buf), &sym, &stat); if (val > 0) { buf[val] = 0; e->compose = strdup(buf); } else e->compose = NULL; } ecore_add_event(ECORE_EVENT_KEY_UP, e, ecore_event_key_up_free); } static void ecore_event_x_handle_button_press(XEvent * xevent) { static Time last_time = 0, last_last_time = 0; static int last_button = 0, last_last_button = 0; static Window last_window = 0, last_last_window = 0; ecore_pointer_xy_set(xevent->xbutton.x_root, xevent->xbutton.y_root); if ((xevent->xbutton.button == 4) || (xevent->xbutton.button == 5)) { Ecore_Event_Wheel *e; e = NEW(Ecore_Event_Wheel, 1); e->win = xevent->xbutton.window; e->root = xevent->xbutton.root; e->x = xevent->xbutton.x; e->y = xevent->xbutton.y; e->rx = xevent->xbutton.x_root; e->ry = xevent->xbutton.y_root; e->time = xevent->xbutton.time; if (xevent->xbutton.button == 5) e->z = 1; else e->z = -1; if (xevent->xbutton.time - last_time < 15) e->z *= 16; else if (xevent->xbutton.time - last_time < 30) e->z *= 4; GETSET_MODS(xevent->xbutton.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_WHEEL, e, ecore_event_generic_free); } else { Ecore_Event_Mouse_Down *e; e = NEW(Ecore_Event_Mouse_Down, 1); e->win = xevent->xbutton.window; e->root = xevent->xbutton.root; e->button = xevent->xbutton.button; e->x = xevent->xbutton.x; e->y = xevent->xbutton.y; e->rx = xevent->xbutton.x_root; e->ry = xevent->xbutton.y_root; e->time = xevent->xbutton.time; e->double_click = 0; e->triple_click = 0; GETSET_MODS(xevent->xbutton.state, e->mods); if (xevent->xbutton.time - last_last_time < 500) { if ((xevent->xbutton.window == (unsigned int)last_window) && (last_window == last_last_window) && (xevent->xbutton.button == (unsigned int)last_button) && (last_button == last_button)) e->triple_click = 1; } else if (xevent->xbutton.time - last_time < 250) { if ((xevent->xbutton.window == (unsigned int)last_window) && (xevent->xbutton.button == (unsigned int)last_button)) e->double_click = 1; } ecore_add_event(ECORE_EVENT_MOUSE_DOWN, e, ecore_event_generic_free); { Ecore_XID *xid = NULL; if (XFindContext(xevent->xbutton.display, e->win, xid_context, (XPointer *) & xid) != XCNOENT) { if ((xid->grab_button_auto_replay) && (xid->grab_button_auto_replay(e))) { ecore_pointer_replay(e->time); } } } } last_last_window = last_window; last_window = xevent->xbutton.window; last_last_button = last_button; last_button = xevent->xbutton.button; last_last_time = last_time; last_time = xevent->xbutton.time; } static void ecore_event_x_handle_button_release(XEvent * xevent) { Ecore_Event_Mouse_Up *e; if (xevent->xbutton.button > 3) return; e = NEW(Ecore_Event_Mouse_Up, 1); e->win = xevent->xbutton.window; e->root = xevent->xbutton.root; e->button = xevent->xbutton.button; e->x = xevent->xbutton.x; e->y = xevent->xbutton.y; e->rx = xevent->xbutton.x_root; e->ry = xevent->xbutton.y_root; e->time = xevent->xbutton.time; GETSET_MODS(xevent->xbutton.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_UP, e, ecore_event_generic_free); } static void ecore_event_x_handle_motion_notify(XEvent * xevent) { Ecore_Event_Mouse_Move *e; ecore_pointer_xy_set(xevent->xmotion.x_root, xevent->xmotion.y_root); e = NEW(Ecore_Event_Mouse_Move, 1); e->win = xevent->xmotion.window; e->root = xevent->xmotion.root; e->x = xevent->xmotion.x; e->y = xevent->xmotion.y; e->rx = xevent->xmotion.x_root; e->ry = xevent->xmotion.y_root; e->time = xevent->xmotion.time; GETSET_MODS(xevent->xmotion.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_MOVE, e, ecore_event_generic_free); } static void ecore_event_x_handle_enter_notify(XEvent * xevent) { Ecore_Event_Window_Enter *e; /* if ((xevent->xcrossing.mode == NotifyGrab) || (xevent->xcrossing.mode == NotifyUngrab)) return;*/ ecore_pointer_xy_set(xevent->xcrossing.x_root, xevent->xcrossing.y_root); e = NEW(Ecore_Event_Window_Enter, 1); e->win = xevent->xcrossing.window; e->root = xevent->xcrossing.root; e->x = xevent->xcrossing.x; e->y = xevent->xcrossing.y; e->rx = xevent->xcrossing.x_root; e->ry = xevent->xcrossing.y_root; e->time = xevent->xcrossing.time; GETSET_MODS(xevent->xcrossing.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_IN, e, ecore_event_generic_free); ecore_window_mouse_set_in(e->win, 1); { Ecore_Event_Mouse_Move *e; e = NEW(Ecore_Event_Mouse_Move, 1); e->win = xevent->xcrossing.window; e->root = xevent->xcrossing.root; e->x = xevent->xcrossing.x; e->y = xevent->xcrossing.y; e->rx = xevent->xcrossing.x_root; e->ry = xevent->xcrossing.y_root; e->time = xevent->xcrossing.time; GETSET_MODS(xevent->xcrossing.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_MOVE, e, ecore_event_generic_free); } } static void ecore_event_x_handle_leave_notify(XEvent * xevent) { Ecore_Event_Window_Leave *e; /* if ((xevent->xcrossing.mode == NotifyGrab) || (xevent->xcrossing.mode == NotifyUngrab)) return;*/ ecore_pointer_xy_set(xevent->xcrossing.x_root, xevent->xcrossing.y_root); { Ecore_Event_Mouse_Move *e; e = NEW(Ecore_Event_Mouse_Move, 1); e->win = xevent->xcrossing.window; e->root = xevent->xcrossing.root; e->x = xevent->xcrossing.x; e->y = xevent->xcrossing.y; e->rx = xevent->xcrossing.x_root; e->ry = xevent->xcrossing.y_root; e->time = xevent->xcrossing.time; GETSET_MODS(xevent->xcrossing.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_MOVE, e, ecore_event_generic_free); } e = NEW(Ecore_Event_Window_Leave, 1); e->win = xevent->xcrossing.window; e->root = xevent->xcrossing.root; e->x = xevent->xcrossing.x; e->y = xevent->xcrossing.y; e->rx = xevent->xcrossing.x_root; e->ry = xevent->xcrossing.y_root; e->time = xevent->xcrossing.time; GETSET_MODS(xevent->xcrossing.state, e->mods); ecore_add_event(ECORE_EVENT_MOUSE_OUT, e, ecore_event_generic_free); ecore_window_mouse_set_in(e->win, 0); } static void ecore_event_x_handle_focus_in(XEvent * xevent) { Ecore_Event_Window_Focus_In *e; e = NEW(Ecore_Event_Window_Focus_In, 1); e->win = xevent->xfocus.window; e->root = ecore_window_get_root(e->win); if (xevent->xfocus.mode != NotifyNormal) e->key_grab = 1; else e->key_grab = 0; ecore_add_event(ECORE_EVENT_WINDOW_FOCUS_IN, e, ecore_event_generic_free); ecore_focus_window_set(e->win); } static void ecore_event_x_handle_focus_out(XEvent * xevent) { Ecore_Event_Window_Focus_Out *e; e = NEW(Ecore_Event_Window_Focus_Out, 1); e->win = xevent->xfocus.window; e->root = ecore_window_get_root(e->win); if (xevent->xfocus.mode != NotifyNormal) e->key_grab = 1; else e->key_grab = 0; ecore_add_event(ECORE_EVENT_WINDOW_FOCUS_OUT, e, ecore_event_generic_free); ecore_focus_window_set(0); } static void ecore_event_x_handle_expose(XEvent * xevent) { Ecore_Event_Window_Expose *e; e = NEW(Ecore_Event_Window_Expose, 1); e->win = xevent->xexpose.window; e->root = ecore_window_get_root(e->win); e->x = xevent->xexpose.x; e->y = xevent->xexpose.y; e->w = xevent->xexpose.width; e->h = xevent->xexpose.height; ecore_add_event(ECORE_EVENT_WINDOW_EXPOSE, e, ecore_event_generic_free); } static void ecore_event_x_handle_visibility_notify(XEvent * xevent) { if (xevent->xvisibility.state != VisibilityPartiallyObscured) { Ecore_Event_Window_Visibility *e; e = NEW(Ecore_Event_Window_Visibility, 1); e->win = xevent->xvisibility.window; e->root = ecore_window_get_root(e->win); if (xevent->xvisibility.state == VisibilityFullyObscured) e->fully_obscured = 1; else e->fully_obscured = 0; ecore_add_event(ECORE_EVENT_WINDOW_VISIBILITY, e, ecore_event_generic_free); } } static void ecore_event_x_handle_create_notify(XEvent * xevent) { Ecore_Event_Window_Create *e; e = NEW(Ecore_Event_Window_Create, 1); e->win = xevent->xcreatewindow.window; ecore_validate_xid(e->win); e->root = ecore_window_get_root(e->win); if (xevent->xcreatewindow.override_redirect) e->override = 1; else e->override = 0; ecore_add_event(ECORE_EVENT_WINDOW_CREATE, e, ecore_event_generic_free); } static void ecore_event_x_handle_destroy_notify(XEvent * xevent) { Ecore_Event_Window_Destroy *e; e = NEW(Ecore_Event_Window_Destroy, 1); e->win = xevent->xdestroywindow.window; e->root = ecore_window_get_root(e->win); ecore_add_event(ECORE_EVENT_WINDOW_DESTROY, e, ecore_event_generic_free); ecore_unvalidate_xid(e->win); } static void ecore_event_x_handle_unmap_notify(XEvent * xevent) { Ecore_Event_Window_Unmap *e; Ecore_XID *xid = NULL; e = NEW(Ecore_Event_Window_Unmap, 1); e->win = xevent->xunmap.window; e->root = ecore_window_get_root(e->win); ecore_add_event(ECORE_EVENT_WINDOW_UNMAP, e, ecore_event_generic_free); xid = ecore_validate_xid(e->win); if (xid) xid->mapped = 0; } static void ecore_event_x_handle_map_notify(XEvent * xevent) { Ecore_Event_Window_Map *e; Ecore_XID *xid = NULL; e = NEW(Ecore_Event_Window_Map, 1); e->win = xevent->xmap.window; e->root = ecore_window_get_root(e->win); ecore_add_event(ECORE_EVENT_WINDOW_MAP, e, ecore_event_generic_free); xid = ecore_validate_xid(e->win); if (xid) xid->mapped = 1; } static void ecore_event_x_handle_map_request(XEvent * xevent) { Ecore_Event_Window_Map_Request *e; e = NEW(Ecore_Event_Window_Map_Request, 1); e->win = xevent->xmaprequest.window; e->root = ecore_window_get_root(e->win); ecore_add_event(ECORE_EVENT_WINDOW_MAP_REQUEST, e, ecore_event_generic_free); ecore_validate_xid(e->win); } static void ecore_event_x_handle_reparent_notify(XEvent * xevent) { Ecore_Event_Window_Reparent *e; Window parent; Ecore_XID *xid = NULL; e = NEW(Ecore_Event_Window_Reparent, 1); e->win = xevent->xreparent.window; xid = ecore_validate_xid(e->win); e->root = ecore_window_get_root(e->win); parent = ecore_window_get_parent(e->win); e->parent_from = parent; e->parent = xevent->xreparent.parent; ecore_validate_xid(e->parent); ecore_del_child(parent, e->win); ecore_add_child(xevent->xreparent.parent, xevent->xreparent.window); if (xid) xid->parent = e->parent; ecore_add_event(ECORE_EVENT_WINDOW_REPARENT, e, ecore_event_generic_free); } static void ecore_event_x_handle_configure_notify(XEvent * xevent) { Ecore_Event_Window_Configure *e; Ecore_XID *xid; e = NEW(Ecore_Event_Window_Configure, 1); e->win = xevent->xconfigure.window; e->root = ecore_window_get_root(e->win); e->x = xevent->xconfigure.x; e->y = xevent->xconfigure.y; e->w = xevent->xconfigure.width; e->h = xevent->xconfigure.height; if (!xevent->xconfigure.send_event) { xid = ecore_validate_xid(e->win); if (xid) { xid->x = e->x; xid->y = e->y; xid->w = e->w; xid->h = e->h; } e->wm_generated = 0; /* FIXME: don't handle redoing stack for xevent->xconfigure.above */ /* member (the window is stacked immediately in stack above this) */ } else e->wm_generated = 1; ecore_add_event(ECORE_EVENT_WINDOW_CONFIGURE, e, ecore_event_generic_free); } static void ecore_event_x_handle_configure_request(XEvent * xevent) { Ecore_Event_Window_Configure_Request *e; e = NEW(Ecore_Event_Window_Configure_Request, 1); e->win = xevent->xconfigurerequest.window; e->root = ecore_window_get_root(e->win); e->x = xevent->xconfigurerequest.x; e->y = xevent->xconfigurerequest.y; e->w = xevent->xconfigurerequest.width; e->h = xevent->xconfigurerequest.height; e->stack_win = xevent->xconfigurerequest.above; e->detail = xevent->xconfigurerequest.detail; e->mask = xevent->xconfigurerequest.value_mask; ecore_add_event(ECORE_EVENT_WINDOW_CONFIGURE_REQUEST, e, ecore_event_generic_free); } static void ecore_event_x_handle_circulate_notify(XEvent * xevent) { Ecore_Event_Window_Circulate *e; e = NEW(Ecore_Event_Window_Circulate, 1); e->win = xevent->xcirculate.window; e->root = ecore_window_get_root(e->win); if (xevent->xcirculate.place == PlaceOnBottom) e->lower = 1; else e->lower = 0; ecore_add_event(ECORE_EVENT_WINDOW_CIRCULATE, e, ecore_event_generic_free); } static void ecore_event_x_handle_circulate_request(XEvent * xevent) { Ecore_Event_Window_Circulate_Request *e; e = NEW(Ecore_Event_Window_Circulate_Request, 1); e->win = xevent->xcirculaterequest.window; e->root = ecore_window_get_root(e->win); if (xevent->xcirculaterequest.place == PlaceOnBottom) e->lower = 1; else e->lower = 0; ecore_add_event(ECORE_EVENT_WINDOW_CIRCULATE_REQUEST, e, ecore_event_generic_free); } static void ecore_event_x_handle_property_notify(XEvent * xevent) { Ecore_Event_Window_Property *e; e = NEW(Ecore_Event_Window_Property, 1); e->win = xevent->xproperty.window; e->root = ecore_window_get_root(e->win); e->atom = xevent->xproperty.atom; e->time = xevent->xproperty.time; ecore_add_event(ECORE_EVENT_WINDOW_PROPERTY, e, ecore_event_generic_free); } static void ecore_event_x_handle_colormap_notify(XEvent * xevent) { Ecore_Event_Colormap *e; e = NEW(Ecore_Event_Colormap, 1); e->win = xevent->xcolormap.window; e->root = ecore_window_get_root(e->win); e->cmap = xevent->xcolormap.colormap; if (xevent->xcolormap.state == ColormapInstalled) e->installed = 1; else e->installed = 0; ecore_add_event(ECORE_EVENT_COLORMAP, e, ecore_event_generic_free); } Ecore_Event_Dnd_Drop_Request *ev_drop_request_pending = NULL; static void ecore_event_x_handle_selection_notify(XEvent * xevent) { Ecore_Event_Dnd_Drop_Request *e; char *data; int size; static Atom atom_xdndactioncopy = 0; static Atom atom_xdndactionmove = 0; static Atom atom_xdndactionlink = 0; static Atom atom_xdndactionask = 0; static Atom atom_xdndactionlist = 0; e = ev_drop_request_pending; if (!e) { Ecore_Event_Paste_Request *e2; e2 = NEW(Ecore_Event_Paste_Request, 1); e2->string = ecore_selection_get_data(xevent->xselection.requestor, xevent->xselection.property); if (e2->string) { e2->win = xevent->xselection.requestor; e2->root = ecore_window_get_root(e2->win); e2->source_win = xevent->xselection.requestor; ecore_add_event(ECORE_EVENT_PASTE_REQUEST, e2, ecore_event_paste_request_free); } else { FREE(e2); } return; } ECORE_ATOM(atom_xdndactioncopy, "XdndActionCopy"); ECORE_ATOM(atom_xdndactionmove, "XdndActionMove"); ECORE_ATOM(atom_xdndactionlink, "XdndActionLink"); ECORE_ATOM(atom_xdndactionask, "XdndActionAsk"); ECORE_ATOM(atom_xdndactionlist, "XdndActionList"); data = ecore_dnd_selection_get(xevent->xany.window, e->source_win, xevent->xselection.property, &size); if (data) { char *s, *buf; int i, is; Atom *method = NULL; method = ecore_window_property_get(e->source_win, atom_xdndactionlist, XA_ATOM, &is); if (method) { e->copy = 0; e->link = 0; e->move = 0; if (*method == atom_xdndactioncopy) e->copy = 1; else if (*method == atom_xdndactionmove) e->move = 1; else if (*method == atom_xdndactionlink) e->link = 1; FREE(method); } else { e->copy = 0; e->link = 0; e->move = 0; } s = data; buf = NEW(char, size); i = 0; is = 0; e->files = NULL; while ((s[is]) && (is < size)) { if ((i == 0) && (s[is] == '#')) { for (; ((s[is] != 0) && (s[is] != '\n')); is++); } else { if (s[is] != '\r') { buf[i++] = s[is]; } else { buf[i] = 0; e->num_files++; REALLOC_PTR(e->files, e->num_files); e->files[e->num_files - 1] = strdup(buf); buf[0] = 0; i = 0; is++; } is++; } } if (i > 0) { buf[i] = 0; e->num_files++; REALLOC_PTR(e->files, e->num_files); e->files[e->num_files - 1] = strdup(buf); } FREE(buf); FREE(data); } ecore_add_event(ECORE_EVENT_DND_DROP_REQUEST, e, ecore_event_dnd_drop_request_free); ev_drop_request_pending = NULL; } static void ecore_event_x_handle_selection_clear(XEvent * xevent) { Ecore_Event_Clear_Selection *e; e = NEW(Ecore_Event_Clear_Selection, 1); e->win = xevent->xselectionclear.window; e->root = ecore_window_get_root(e->win); e->selection = xevent->xselectionclear.selection; ecore_add_event(ECORE_EVENT_CLEAR_SELECTION, e, ecore_event_generic_free); } static void ecore_event_x_handle_selection_request(XEvent * xevent) { static Atom atom_xdndselection = 0; static Atom atom_text_plain = 0; static Atom atom_text_uri_list = 0; static Atom atom_text_moz_url = 0; static Atom atom_netscape_url = 0; static Atom atom_text_selection = 0; Ecore_Event_Dnd_Data_Request *e; ECORE_ATOM(atom_xdndselection, "XdndSelection"); ECORE_ATOM(atom_text_plain, "text/plain"); ECORE_ATOM(atom_text_uri_list, "text/uri-list"); ECORE_ATOM(atom_text_moz_url, "text/x-moz-url"); ECORE_ATOM(atom_netscape_url, "_NETSCAPE_URL"); ECORE_ATOM(atom_text_selection, "TEXT_SELECTION"); if (xevent->xselectionrequest.selection == atom_xdndselection) { e = NEW(Ecore_Event_Dnd_Data_Request, 1); e->win = xevent->xselectionrequest.owner; e->root = ecore_window_get_root(e->win); e->source_win = xevent->xselectionrequest.requestor; e->plain_text =0; e->uri_list = 0; e->moz_url = 0; e->netscape_url = 0; if (xevent->xselectionrequest.target == atom_text_plain) e->plain_text = 1; if (xevent->xselectionrequest.target == atom_text_uri_list) e->uri_list = 1; if (xevent->xselectionrequest.target == atom_text_moz_url) e->moz_url = 1; if (xevent->xselectionrequest.target == atom_netscape_url) e->netscape_url = 1; e->destination_atom = xevent->xselectionrequest.property; ecore_add_event(ECORE_EVENT_DND_DATA_REQUEST, e, ecore_event_generic_free); } else { XEvent ev; Atom target_list[2]; static Atom xa_targets = None; if (xa_targets == None) xa_targets = XInternAtom(xevent->xselectionrequest.display, "TARGETS", False); ev.xselection.type = SelectionNotify; ev.xselection.property = None; ev.xselection.display = xevent->xselectionrequest.display; ev.xselection.requestor = xevent->xselectionrequest.requestor; ev.xselection.selection = xevent->xselectionrequest.selection; ev.xselection.target = xevent->xselectionrequest.target; ev.xselection.time = xevent->xselectionrequest.time; if (xevent->xselectionrequest.target == xa_targets) { target_list[0] = (Atom) xa_targets; target_list[1] = (Atom) XA_STRING; XChangeProperty(xevent->xselectionrequest.display, xevent->xselectionrequest.requestor, xevent->xselectionrequest.property, xevent->xselectionrequest.target, (8 * sizeof(target_list[0])), PropModeReplace, (unsigned char *)target_list, (sizeof(target_list) / sizeof(target_list[0]))); ev.xselection.property = xevent->xselectionrequest.property; } else if (xevent->xselectionrequest.target == XA_STRING) { void *data; int size; data = ecore_window_property_get(xevent->xselectionrequest.owner, atom_text_selection, XA_STRING, &size); if (data) { XChangeProperty(xevent->xselectionrequest.display, xevent->xselectionrequest.requestor, xevent->xselectionrequest.property, xevent->xselectionrequest.target, 8, PropModeReplace, data, size); FREE(data); } ev.xselection.property = xevent->xselectionrequest.property; } XSendEvent(xevent->xselectionrequest.display, xevent->xselectionrequest.requestor, False, 0, &ev); } } static void ecore_event_x_handle_client_message(XEvent * xevent) { static Atom atom_wm_delete_window = 0; static Atom atom_wm_protocols = 0; static Atom atom_xdndstatus = 0; static Atom atom_xdndenter = 0; static Atom atom_xdndleave = 0; static Atom atom_xdndfinished = 0; static Atom atom_xdndposition = 0; static Atom atom_xdnddrop = 0; static Atom atom_text_uri_list = 0; static Atom atom_xdndactioncopy = 0; static Atom atom_xdndactionlink = 0; static Atom atom_xdndactionmove = 0; static Atom atom_xdndactionprivate = 0; /* setup some known atoms to translate this message into a sensible event */ ECORE_ATOM(atom_wm_delete_window, "WM_DELETE_WINDOW"); ECORE_ATOM(atom_wm_protocols, "WM_PROTOCOLS"); ECORE_ATOM(atom_xdndstatus, "XdndStatus"); ECORE_ATOM(atom_xdndfinished, "XdndFinished"); ECORE_ATOM(atom_xdndenter, "XdndEnter"); ECORE_ATOM(atom_xdndleave, "XdndLeave"); ECORE_ATOM(atom_xdnddrop, "XdndDrop"); ECORE_ATOM(atom_xdndposition, "XdndPosition"); ECORE_ATOM(atom_xdndactioncopy, "XdndActionCopy"); ECORE_ATOM(atom_xdndactionlink, "XdndActionLink"); ECORE_ATOM(atom_xdndactionmove, "XdndActionMove"); ECORE_ATOM(atom_xdndactionprivate, "XdndActionPrivate"); ECORE_ATOM(atom_text_uri_list, "text/uri-list"); /* first type = delete event sent to client */ if ((xevent->xclient.message_type == atom_wm_protocols) && (xevent->xclient.format == 32) && (xevent->xclient.data.l[0] == (long)atom_wm_delete_window)) { Ecore_Event_Window_Delete *e; e = NEW(Ecore_Event_Window_Delete, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); ecore_add_event(ECORE_EVENT_WINDOW_DELETE, e, ecore_event_generic_free); } else if ((xevent->xclient.message_type == atom_xdndenter) && (xevent->xclient.format == 32)) { /* if ((xevent->xclient.data.l[2] == (long)atom_text_uri_list) || * (xevent->xclient.data.l[3] == (long)atom_text_uri_list) || * (xevent->xclient.data.l[4] == (long)atom_text_uri_list)) */ { Ecore_Event_Dnd_Drop_Request *e; if (ev_drop_request_pending) { ecore_event_dnd_drop_request_free(ev_drop_request_pending); ev_drop_request_pending = NULL; } e = NEW(Ecore_Event_Dnd_Drop_Request, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); e->source_win = (Window) xevent->xclient.data.l[0]; if (!ecore_dnd_selection_convert (e->source_win, e->win, atom_text_uri_list)) { FREE(e); return; } e->files = NULL; e->num_files = 0; ev_drop_request_pending = e; } } else if ((xevent->xclient.message_type == atom_xdndleave) && (xevent->xclient.format == 32)) { Ecore_Event_Dnd_Drop_End *e; e = NEW(Ecore_Event_Dnd_Drop_End, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); e->source_win = (Window) xevent->xclient.data.l[0]; ecore_add_event(ECORE_EVENT_DND_DROP_END, e, ecore_event_generic_free); } else if ((xevent->xclient.message_type == atom_xdndposition) && (xevent->xclient.format == 32)) { Ecore_Event_Dnd_Drop_Position *e; e = NEW(Ecore_Event_Dnd_Drop_Position, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); e->source_win = (Window) xevent->xclient.data.l[0]; e->x = (xevent->xclient.data.l[2] >> 16) & 0xffff; e->y = xevent->xclient.data.l[2] & 0xffff; ecore_add_event(ECORE_EVENT_DND_DROP_POSITION, e, ecore_event_generic_free); } else if ((xevent->xclient.message_type == atom_xdndstatus) && (xevent->xclient.format == 32)) { Ecore_Event_Dnd_Drop_Status *e; ecore_clear_target_status(); e = NEW(Ecore_Event_Dnd_Drop_Status, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); e->source_win = (Window) xevent->xclient.data.l[0]; e->x = (xevent->xclient.data.l[2] >> 16) & 0xffff; e->y = xevent->xclient.data.l[2] & 0xffff; e->w = (xevent->xclient.data.l[3] >> 16) & 0xffff; e->h = xevent->xclient.data.l[3] & 0xffff; e->copy = e->link = e->move = e->e_private = 0; if( xevent->xclient.data.l[4] == atom_xdndactioncopy ) e->copy = 1; else if( xevent->xclient.data.l[4] == atom_xdndactionlink ) e->link = 1; else if( xevent->xclient.data.l[4] == atom_xdndactionmove ) e->move = 1; else if( xevent->xclient.data.l[4] == atom_xdndactionprivate ) e->e_private = 1; if (xevent->xclient.data.l[1] & 0x1) e->ok = 1; else e->ok = 0; if (xevent->xclient.data.l[1] & 0x2) e->all_position_msgs = 1; else e->all_position_msgs = 0; ecore_add_event(ECORE_EVENT_DND_DROP_STATUS, e, ecore_event_generic_free); } else if ((xevent->xclient.message_type == atom_xdndfinished) && (xevent->xclient.format == 32)) { Ecore_Event_Dnd_Drop_End *e; e = NEW(Ecore_Event_Dnd_Drop_End, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); e->source_win = (Window) xevent->xclient.data.l[0]; ecore_add_event(ECORE_EVENT_DND_DROP_END, e, ecore_event_generic_free); } else if ((xevent->xclient.message_type == atom_xdnddrop) && (xevent->xclient.format == 32)) { Ecore_Event_Dnd_Drop *e; e = NEW(Ecore_Event_Dnd_Drop, 1); e->win = xevent->xclient.window; e->root = ecore_window_get_root(e->win); e->source_win = (Window) xevent->xclient.data.l[0]; ecore_add_event(ECORE_EVENT_DND_DROP, e, ecore_event_generic_free); } else { Ecore_Event_Message *e; e = NEW(Ecore_Event_Message, 1); e->win = xevent->xclient.window; e->format = xevent->xclient.format; e->atom = xevent->xclient.message_type; MEMCPY(xevent->xclient.data.b, e->data.b, char, 20); ecore_add_event(ECORE_EVENT_MESSAGE, e, ecore_event_generic_free); } } static void ecore_event_x_handle_shape_change(XEvent * xevent) { Ecore_Event_Window_Shape *e; XShapeEvent *shape_event; shape_event = (XShapeEvent *) xevent; e = NEW(Ecore_Event_Window_Shape, 1); e->win = shape_event->window; e->root = ecore_window_get_root(e->win); e->time = shape_event->time; ecore_add_event(ECORE_EVENT_WINDOW_SHAPE, e, ecore_event_generic_free); } char * ecore_keypress_translate_into_typeable(Ecore_Event_Key_Down * e) { /* exceptions */ if ((!strcmp(e->key, "Delete")) || (!strcmp(e->key, "BackSpace")) || (!strcmp(e->key, "Tab")) || (!strcmp(e->key, "Escape")) || (!strcmp(e->key, "Return")) || (!strcmp(e->key, "KP_Enter")) || (!strcmp(e->key, "Enter")) || (!strcmp(e->key, "KP_Divide")) || (!strcmp(e->key, "KP_Multiply")) || (!strcmp(e->key, "KP_Subtract")) || (!strcmp(e->key, "KP_Add")) || (!strcmp(e->key, "Enter"))) return NULL; return e->compose; }