forked from enlightenment/enlightenment
4273 lines
116 KiB
C
4273 lines
116 KiB
C
/*
|
|
* vim:ts=8:sw=3:sts=8:noexpandtab:cino=>5n-3f0^-2{2
|
|
*/
|
|
#include "e.h"
|
|
|
|
//#define INOUTDEBUG_MOUSE 1
|
|
//#define INOUTDEBUG_FOCUS 1
|
|
|
|
#define RESIZE_NONE 0
|
|
#define RESIZE_TL 1
|
|
#define RESIZE_T 2
|
|
#define RESIZE_TR 3
|
|
#define RESIZE_R 4
|
|
#define RESIZE_BR 5
|
|
#define RESIZE_B 6
|
|
#define RESIZE_BL 7
|
|
#define RESIZE_L 8
|
|
|
|
/* local subsystem functions */
|
|
static void _e_border_free(E_Border *bd);
|
|
static void _e_border_del(E_Border *bd);
|
|
|
|
/* FIXME: these likely belong in a separate icccm/client handler */
|
|
/* and the border needs to become a dumb object that just does what its */
|
|
/* told to do */
|
|
static int _e_border_cb_window_show_request(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_destroy(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_hide(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_reparent(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_configure_request(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_resize_request(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_gravity(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_stack_request(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_property(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_colormap(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_shape(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_focus_in(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_focus_out(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_window_state(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_desktop_change(void *data, int ev_type, void *ev);
|
|
static int _e_border_cb_client_message(void *data, int ev_type, void *ev);
|
|
|
|
static void _e_border_cb_signal_move_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_move_stop(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_tl_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_t_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_tr_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_r_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_br_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_b_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_bl_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_l_start(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_resize_stop(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_action(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static void _e_border_cb_signal_drag(void *data, Evas_Object *obj, const char *emission, const char *source);
|
|
static int _e_border_cb_mouse_in(void *data, int type, void *event);
|
|
static int _e_border_cb_mouse_out(void *data, int type, void *event);
|
|
static int _e_border_cb_mouse_down(void *data, int type, void *event);
|
|
static int _e_border_cb_mouse_up(void *data, int type, void *event);
|
|
static int _e_border_cb_mouse_move(void *data, int type, void *event);
|
|
static int _e_border_cb_mouse_wheel(void *data, int type, void *event);
|
|
static int _e_border_cb_grab_replay(void *data, int type, void *event);
|
|
|
|
static void _e_border_eval(E_Border *bd);
|
|
static void _e_border_resize_limit(E_Border *bd, int *w, int *h);
|
|
static void _e_border_moveinfo_gather(E_Border *bd, const char *source);
|
|
static void _e_border_resize_handle(E_Border *bd);
|
|
|
|
static int _e_border_shade_animator(void *data);
|
|
|
|
static void _e_border_cb_border_menu_end(void *data, E_Menu *m);
|
|
static void _e_border_menu_show(E_Border *bd, Evas_Coord x, Evas_Coord y, int key);
|
|
static void _e_border_menu_cb_close(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_cb_iconify(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_cb_maximize(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_cb_shade(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_cb_icon_edit(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_cb_stick(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_cb_on_top(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_sendto_pre_cb(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
static void _e_border_menu_sendto_cb(void *data, E_Menu *m, E_Menu_Item *mi);
|
|
|
|
static void _e_border_event_border_add_free(void *data, void *ev);
|
|
static void _e_border_event_border_remove_free(void *data, void *ev);
|
|
static void _e_border_event_border_zone_set_free(void *data, void *ev);
|
|
static void _e_border_event_border_desk_set_free(void *data, void *ev);
|
|
static void _e_border_event_border_raise_free(void *data, void *ev);
|
|
static void _e_border_event_border_lower_free(void *data, void *ev);
|
|
static void _e_border_event_border_icon_change_free(void *data, void *ev);
|
|
static void _e_border_event_border_resize_free(void *data, void *ev);
|
|
static void _e_border_event_border_move_free(void *data, void *ev);
|
|
static void _e_border_event_border_show_free(void *data, void *ev);
|
|
static void _e_border_event_border_hide_free(void *data, void *ev);
|
|
static void _e_border_event_border_iconify_free(void *data, void *ev);
|
|
static void _e_border_event_border_uniconify_free(void *data, void *ev);
|
|
static void _e_border_event_border_stick_free(void *data, void *ev);
|
|
static void _e_border_event_border_unstick_free(void *data, void *ev);
|
|
|
|
static void _e_border_zone_update(E_Border *bd);
|
|
static void _e_border_desk_update(E_Border *bd);
|
|
|
|
static void _e_border_resize_begin(E_Border *bd);
|
|
static void _e_border_resize_end(E_Border *bd);
|
|
static void _e_border_resize_update(E_Border *bd);
|
|
|
|
static void _e_border_move_begin(E_Border *bd);
|
|
static void _e_border_move_end(E_Border *bd);
|
|
static void _e_border_move_update(E_Border *bd);
|
|
|
|
static int _e_border_cb_focus_fix(void *data);
|
|
|
|
/* local subsystem globals */
|
|
static Evas_List *handlers = NULL;
|
|
static Evas_List *borders = NULL;
|
|
static E_Border *focused = NULL;
|
|
|
|
static E_Border *resize = NULL;
|
|
static E_Border *move = NULL;
|
|
|
|
static Ecore_Timer *focus_fix_timer = NULL;
|
|
|
|
int E_EVENT_BORDER_ADD = 0;
|
|
int E_EVENT_BORDER_REMOVE = 0;
|
|
int E_EVENT_BORDER_ZONE_SET = 0;
|
|
int E_EVENT_BORDER_DESK_SET = 0;
|
|
int E_EVENT_BORDER_RESIZE = 0;
|
|
int E_EVENT_BORDER_MOVE = 0;
|
|
int E_EVENT_BORDER_SHOW = 0;
|
|
int E_EVENT_BORDER_HIDE = 0;
|
|
int E_EVENT_BORDER_ICONIFY = 0;
|
|
int E_EVENT_BORDER_UNICONIFY = 0;
|
|
int E_EVENT_BORDER_STICK = 0;
|
|
int E_EVENT_BORDER_UNSTICK = 0;
|
|
int E_EVENT_BORDER_RAISE = 0;
|
|
int E_EVENT_BORDER_LOWER = 0;
|
|
int E_EVENT_BORDER_ICON_CHANGE = 0;
|
|
|
|
#define GRAV_SET(bd, grav) \
|
|
ecore_x_window_gravity_set(bd->bg_win, grav); \
|
|
ecore_x_window_gravity_set(bd->client.shell_win, grav); \
|
|
ecore_x_window_gravity_set(bd->client.win, grav); \
|
|
ecore_x_window_gravity_set(bd->bg_subwin, grav); \
|
|
ecore_x_window_pixel_gravity_set(bd->bg_subwin, grav);
|
|
|
|
/* externally accessible functions */
|
|
int
|
|
e_border_init(void)
|
|
{
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_SHOW_REQUEST, _e_border_cb_window_show_request, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_DESTROY, _e_border_cb_window_destroy, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_HIDE, _e_border_cb_window_hide, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_REPARENT, _e_border_cb_window_reparent, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_CONFIGURE_REQUEST, _e_border_cb_window_configure_request, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_RESIZE_REQUEST, _e_border_cb_window_resize_request, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_GRAVITY, _e_border_cb_window_gravity, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_STACK_REQUEST, _e_border_cb_window_stack_request, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_PROPERTY, _e_border_cb_window_property, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_COLORMAP, _e_border_cb_window_colormap, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_SHAPE, _e_border_cb_window_shape, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_FOCUS_IN, _e_border_cb_window_focus_in, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_FOCUS_OUT, _e_border_cb_window_focus_out, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_STATE, _e_border_cb_window_state, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_DESKTOP_CHANGE, _e_border_cb_desktop_change, NULL));
|
|
handlers = evas_list_append(handlers, ecore_event_handler_add(ECORE_X_EVENT_CLIENT_MESSAGE, _e_border_cb_client_message, NULL));
|
|
ecore_x_passive_grab_replay_func_set(_e_border_cb_grab_replay, NULL);
|
|
|
|
E_EVENT_BORDER_ADD = ecore_event_type_new();
|
|
E_EVENT_BORDER_REMOVE = ecore_event_type_new();
|
|
E_EVENT_BORDER_DESK_SET = ecore_event_type_new();
|
|
E_EVENT_BORDER_ZONE_SET = ecore_event_type_new();
|
|
E_EVENT_BORDER_RESIZE = ecore_event_type_new();
|
|
E_EVENT_BORDER_MOVE = ecore_event_type_new();
|
|
E_EVENT_BORDER_SHOW = ecore_event_type_new();
|
|
E_EVENT_BORDER_HIDE = ecore_event_type_new();
|
|
E_EVENT_BORDER_ICONIFY = ecore_event_type_new();
|
|
E_EVENT_BORDER_UNICONIFY = ecore_event_type_new();
|
|
E_EVENT_BORDER_STICK = ecore_event_type_new();
|
|
E_EVENT_BORDER_UNSTICK = ecore_event_type_new();
|
|
E_EVENT_BORDER_RAISE = ecore_event_type_new();
|
|
E_EVENT_BORDER_LOWER = ecore_event_type_new();
|
|
E_EVENT_BORDER_ICON_CHANGE = ecore_event_type_new();
|
|
|
|
focus_fix_timer = ecore_timer_add(0.1, _e_border_cb_focus_fix, NULL);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int
|
|
e_border_shutdown(void)
|
|
{
|
|
while (handlers)
|
|
{
|
|
Ecore_Event_Handler *h;
|
|
|
|
h = handlers->data;
|
|
handlers = evas_list_remove_list(handlers, handlers);
|
|
ecore_event_handler_del(h);
|
|
}
|
|
ecore_timer_del(focus_fix_timer);
|
|
focus_fix_timer = NULL;
|
|
return 1;
|
|
}
|
|
|
|
E_Border *
|
|
e_border_new(E_Container *con, Ecore_X_Window win, int first_map)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Window_Attributes *att;
|
|
unsigned int managed, desk[2];
|
|
int deskx, desky;
|
|
|
|
bd = E_OBJECT_ALLOC(E_Border, E_BORDER_TYPE, _e_border_free);
|
|
if (!bd) return NULL;
|
|
e_object_del_func_set(E_OBJECT(bd), E_OBJECT_CLEANUP_FUNC(_e_border_del));
|
|
|
|
printf("##- NEW CLIENT 0x%x\n", win);
|
|
bd->w = 1;
|
|
bd->h = 1;
|
|
bd->win = ecore_x_window_override_new(con->win, 0, 0, bd->w, bd->h);
|
|
ecore_x_window_shape_events_select(bd->win, 1);
|
|
e_bindings_mouse_grab(E_BINDING_CONTEXT_BORDER, bd->win);
|
|
if (e_canvas_engine_decide(e_config->evas_engine_borders) ==
|
|
E_EVAS_ENGINE_GL_X11)
|
|
{
|
|
bd->bg_ecore_evas = ecore_evas_gl_x11_new(NULL, bd->win,
|
|
0, 0, bd->w, bd->h);
|
|
ecore_evas_gl_x11_direct_resize_set(bd->bg_ecore_evas, 1);
|
|
bd->bg_win = ecore_evas_gl_x11_window_get(bd->bg_ecore_evas);
|
|
bd->bg_subwin = ecore_evas_gl_x11_subwindow_get(bd->bg_ecore_evas);
|
|
}
|
|
else
|
|
{
|
|
bd->bg_ecore_evas = ecore_evas_software_x11_new(NULL, bd->win,
|
|
0, 0, bd->w, bd->h);
|
|
ecore_evas_software_x11_direct_resize_set(bd->bg_ecore_evas, 1);
|
|
bd->bg_win = ecore_evas_software_x11_window_get(bd->bg_ecore_evas);
|
|
bd->bg_subwin = ecore_evas_software_x11_subwindow_get(bd->bg_ecore_evas);
|
|
}
|
|
e_canvas_add(bd->bg_ecore_evas);
|
|
bd->event_win = ecore_x_window_input_new(bd->win, 0, 0, bd->w, bd->h);
|
|
bd->bg_evas = ecore_evas_get(bd->bg_ecore_evas);
|
|
ecore_x_window_shape_events_select(bd->bg_win, 1);
|
|
ecore_evas_name_class_set(bd->bg_ecore_evas, "E", "Frame_Window");
|
|
ecore_evas_title_set(bd->bg_ecore_evas, "Enlightenment Frame");
|
|
bd->client.shell_win = ecore_x_window_override_new(bd->win, 0, 0, 1, 1);
|
|
ecore_x_window_container_manage(bd->client.shell_win);
|
|
ecore_x_window_client_manage(win);
|
|
/* FIXME: Round trip. XCB */
|
|
/* fetch needed to avoid grabbing the server as window may vanish */
|
|
att = &bd->client.initial_attributes;
|
|
if ((!ecore_x_window_attributes_get(win, att)) || (att->input_only))
|
|
{
|
|
printf("##- ATTR FETCH FAILED/INPUT ONLY FOR 0x%x - ABORT MANAGE\n", win);
|
|
e_canvas_del(bd->bg_ecore_evas);
|
|
ecore_evas_free(bd->bg_ecore_evas);
|
|
ecore_x_window_del(bd->client.shell_win);
|
|
e_bindings_mouse_ungrab(E_BINDING_CONTEXT_BORDER, bd->win);
|
|
ecore_x_window_del(bd->win);
|
|
free(bd);
|
|
return NULL;
|
|
}
|
|
bd->handlers = evas_list_append(bd->handlers, ecore_event_handler_add(ECORE_X_EVENT_MOUSE_IN, _e_border_cb_mouse_in, bd));
|
|
bd->handlers = evas_list_append(bd->handlers, ecore_event_handler_add(ECORE_X_EVENT_MOUSE_OUT, _e_border_cb_mouse_out, bd));
|
|
bd->handlers = evas_list_append(bd->handlers, ecore_event_handler_add(ECORE_X_EVENT_MOUSE_BUTTON_DOWN, _e_border_cb_mouse_down, bd));
|
|
bd->handlers = evas_list_append(bd->handlers, ecore_event_handler_add(ECORE_X_EVENT_MOUSE_BUTTON_UP, _e_border_cb_mouse_up, bd));
|
|
bd->handlers = evas_list_append(bd->handlers, ecore_event_handler_add(ECORE_X_EVENT_MOUSE_MOVE, _e_border_cb_mouse_move, bd));
|
|
bd->handlers = evas_list_append(bd->handlers, ecore_event_handler_add(ECORE_X_EVENT_MOUSE_WHEEL, _e_border_cb_mouse_wheel, bd));
|
|
|
|
bd->client.win = win;
|
|
|
|
bd->client.icccm.title = strdup("");
|
|
bd->client.icccm.name = strdup("");
|
|
bd->client.icccm.class = strdup("");
|
|
bd->client.icccm.icon_name = strdup("");
|
|
bd->client.icccm.machine = strdup("");
|
|
bd->client.icccm.min_w = 1;
|
|
bd->client.icccm.min_h = 1;
|
|
bd->client.icccm.max_w = 32767;
|
|
bd->client.icccm.max_h = 32767;
|
|
bd->client.icccm.base_w = 0;
|
|
bd->client.icccm.base_h = 0;
|
|
bd->client.icccm.step_w = 1;
|
|
bd->client.icccm.step_h = 1;
|
|
bd->client.icccm.min_aspect = 0.0;
|
|
bd->client.icccm.max_aspect = 0.0;
|
|
bd->client.icccm.accepts_focus = 1;
|
|
|
|
bd->client.icccm.fetch.title = 1;
|
|
bd->client.icccm.fetch.name_class = 1;
|
|
bd->client.icccm.fetch.icon_name = 1;
|
|
bd->client.icccm.fetch.machine = 1;
|
|
bd->client.icccm.fetch.hints = 1;
|
|
bd->client.icccm.fetch.size_pos_hints = 1;
|
|
bd->client.icccm.fetch.protocol = 1;
|
|
bd->client.mwm.fetch.hints = 1;
|
|
bd->client.border.changed = 1;
|
|
|
|
bd->client.w = att->w;
|
|
bd->client.h = att->h;
|
|
|
|
bd->w = bd->client.w;
|
|
bd->h = bd->client.h;
|
|
|
|
bd->layer = 100;
|
|
bd->changes.size = 1;
|
|
bd->changes.shape = 1;
|
|
|
|
printf("##- ON MAP CLIENT 0x%x SIZE %ix%i\n",
|
|
bd->client.win, bd->client.w, bd->client.h);
|
|
|
|
/* FIXME: if first_map is 1 then we should ignore the first hide event
|
|
* or ensure the window is alreayd hidden and events flushed before we
|
|
* create a border for it */
|
|
if (first_map)
|
|
{
|
|
printf("##- FIRST MAP\n");
|
|
bd->x = att->x;
|
|
bd->y = att->y;
|
|
bd->changes.pos = 1;
|
|
bd->re_manage = 1;
|
|
bd->ignore_first_unmap = 2;
|
|
}
|
|
|
|
ecore_x_window_save_set_add(win);
|
|
ecore_x_window_reparent(win, bd->client.shell_win, 0, 0);
|
|
ecore_x_window_border_width_set(win, 0);
|
|
ecore_x_window_show(bd->event_win);
|
|
ecore_x_window_show(bd->client.shell_win);
|
|
bd->shape = e_container_shape_add(con);
|
|
|
|
bd->new_client = 1;
|
|
bd->changed = 1;
|
|
|
|
bd->zone = e_zone_current_get(con);
|
|
bd->desk = e_desk_current_get(bd->zone);
|
|
e_container_border_add(bd);
|
|
borders = evas_list_append(borders, bd);
|
|
|
|
managed = 1;
|
|
ecore_x_window_prop_card32_set(win, E_ATOM_MANAGED, &managed, 1);
|
|
ecore_x_window_prop_card32_set(win, E_ATOM_CONTAINER, &bd->zone->container->num, 1);
|
|
ecore_x_window_prop_card32_set(win, E_ATOM_ZONE, &bd->zone->num, 1);
|
|
e_desk_xy_get(bd->desk, &deskx, &desky);
|
|
desk[0] = deskx;
|
|
desk[1] = desky;
|
|
ecore_x_window_prop_card32_set(win, E_ATOM_DESK, desk, 2);
|
|
|
|
return bd;
|
|
}
|
|
|
|
void
|
|
e_border_zone_set(E_Border *bd, E_Zone *zone)
|
|
{
|
|
E_Event_Border_Zone_Set *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
E_OBJECT_CHECK(zone);
|
|
E_OBJECT_TYPE_CHECK(zone, E_ZONE_TYPE);
|
|
if (bd->zone == zone) return;
|
|
bd->zone = zone;
|
|
|
|
if (bd->desk->zone != bd->zone)
|
|
{
|
|
e_border_desk_set(bd, e_desk_current_get(bd->zone));
|
|
}
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Zone_Set));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ev->zone = zone;
|
|
e_object_ref(E_OBJECT(zone));
|
|
ecore_event_add(E_EVENT_BORDER_ZONE_SET, ev, _e_border_event_border_zone_set_free, NULL);
|
|
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_ZONE, &bd->zone->num, 1);
|
|
}
|
|
|
|
void
|
|
e_border_desk_set(E_Border *bd, E_Desk *desk)
|
|
{
|
|
E_Event_Border_Desk_Set *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
E_OBJECT_CHECK(desk);
|
|
E_OBJECT_TYPE_CHECK(desk, E_DESK_TYPE);
|
|
if (bd->desk == desk) return;
|
|
bd->desk = desk;
|
|
e_border_zone_set(bd, desk->zone);
|
|
|
|
e_hints_window_desktop_set(bd);
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Desk_Set));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ev->desk = desk;
|
|
e_object_ref(E_OBJECT(desk));
|
|
ecore_event_add(E_EVENT_BORDER_DESK_SET, ev, _e_border_event_border_desk_set_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_show(E_Border *bd)
|
|
{
|
|
E_Event_Border_Show *ev;
|
|
unsigned int visible;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (bd->visible) return;
|
|
e_container_shape_show(bd->shape);
|
|
ecore_x_window_show(bd->client.win);
|
|
e_hints_window_visible_set(bd);
|
|
bd->visible = 1;
|
|
bd->changes.visible = 1;
|
|
|
|
visible = 1;
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_MAPPED, &visible, 1);
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_MANAGED, &visible, 1);
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Show));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_SHOW, ev, _e_border_event_border_show_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_hide(E_Border *bd, int manage)
|
|
{
|
|
unsigned int visible;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (!bd->visible) return;
|
|
printf("hide1\n");
|
|
if (bd->moving) return;
|
|
printf("hide2\nn");
|
|
|
|
ecore_x_window_hide(bd->client.win);
|
|
e_container_shape_hide(bd->shape);
|
|
if (!bd->iconic)
|
|
e_hints_window_hidden_set(bd);
|
|
|
|
bd->visible = 0;
|
|
bd->changes.visible = 1;
|
|
|
|
visible = 0;
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_MAPPED, &visible, 1);
|
|
if (!manage)
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_MANAGED, &visible, 1);
|
|
|
|
{
|
|
E_Event_Border_Hide *ev;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Hide));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
printf("ref bd %p to %i\n", bd, e_object_ref_get(E_OBJECT(bd)));
|
|
ecore_event_add(E_EVENT_BORDER_HIDE, ev, _e_border_event_border_hide_free, NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_move(E_Border *bd, int x, int y)
|
|
{
|
|
E_Event_Border_Move *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (bd->new_client)
|
|
{
|
|
E_Border_Pending_Move_Resize *pnd;
|
|
|
|
pnd = E_NEW(E_Border_Pending_Move_Resize, 1);
|
|
if (!pnd) return;
|
|
pnd->move = 1;
|
|
pnd->x = x;
|
|
pnd->y = y;
|
|
bd->pending_move_resize = evas_list_append(bd->pending_move_resize, pnd);
|
|
return;
|
|
}
|
|
if ((x == bd->x) && (y == bd->y)) return;
|
|
bd->x = x;
|
|
bd->y = y;
|
|
bd->changed = 1;
|
|
bd->changes.pos = 1;
|
|
ecore_x_icccm_move_resize_send(bd->client.win,
|
|
bd->x + bd->client_inset.l,
|
|
bd->y + bd->client_inset.t,
|
|
bd->client.w,
|
|
bd->client.h);
|
|
_e_border_move_update(bd);
|
|
_e_border_zone_update(bd);
|
|
_e_border_desk_update(bd);
|
|
ev = calloc(1, sizeof(E_Event_Border_Move));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_MOVE, ev, _e_border_event_border_move_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_resize(E_Border *bd, int w, int h)
|
|
{
|
|
E_Event_Border_Resize *ev;
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (bd->new_client)
|
|
{
|
|
E_Border_Pending_Move_Resize *pnd;
|
|
|
|
pnd = E_NEW(E_Border_Pending_Move_Resize, 1);
|
|
if (!pnd) return;
|
|
pnd->resize = 1;
|
|
pnd->w = w;
|
|
pnd->h = h;
|
|
bd->pending_move_resize = evas_list_append(bd->pending_move_resize, pnd);
|
|
return;
|
|
}
|
|
if ((w == bd->w) && (h == bd->h)) return;
|
|
bd->w = w;
|
|
bd->h = h;
|
|
bd->client.w = bd->w - (bd->client_inset.l + bd->client_inset.r);
|
|
bd->client.h = bd->h - (bd->client_inset.t + bd->client_inset.b);
|
|
bd->changed = 1;
|
|
bd->changes.size = 1;
|
|
ecore_x_icccm_move_resize_send(bd->client.win,
|
|
bd->x + bd->client_inset.l,
|
|
bd->y + bd->client_inset.t,
|
|
bd->client.w,
|
|
bd->client.h);
|
|
_e_border_zone_update(bd);
|
|
ev = calloc(1, sizeof(E_Event_Border_Resize));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_RESIZE, ev, _e_border_event_border_resize_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_move_resize(E_Border *bd, int x, int y, int w, int h)
|
|
{
|
|
E_Event_Border_Move *mev;
|
|
E_Event_Border_Resize *rev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (bd->new_client)
|
|
{
|
|
E_Border_Pending_Move_Resize *pnd;
|
|
|
|
pnd = E_NEW(E_Border_Pending_Move_Resize, 1);
|
|
if (!pnd) return;
|
|
pnd->move = 1;
|
|
pnd->resize = 1;
|
|
pnd->x = x;
|
|
pnd->y = y;
|
|
pnd->w = w;
|
|
pnd->h = h;
|
|
bd->pending_move_resize = evas_list_append(bd->pending_move_resize, pnd);
|
|
return;
|
|
}
|
|
if ((x == bd->x) && (y == bd->y) && (w == bd->w) && (h == bd->h)) return;
|
|
bd->x = x;
|
|
bd->y = y;
|
|
bd->w = w;
|
|
bd->h = h;
|
|
bd->client.w = bd->w - (bd->client_inset.l + bd->client_inset.r);
|
|
bd->client.h = bd->h - (bd->client_inset.t + bd->client_inset.b);
|
|
bd->changed = 1;
|
|
bd->changes.pos = 1;
|
|
bd->changes.size = 1;
|
|
ecore_x_icccm_move_resize_send(bd->client.win,
|
|
bd->x + bd->client_inset.l,
|
|
bd->y + bd->client_inset.t,
|
|
bd->client.w,
|
|
bd->client.h);
|
|
_e_border_resize_update(bd);
|
|
_e_border_zone_update(bd);
|
|
mev = calloc(1, sizeof(E_Event_Border_Move));
|
|
mev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_MOVE, mev, _e_border_event_border_move_free, NULL);
|
|
|
|
rev = calloc(1, sizeof(E_Event_Border_Resize));
|
|
rev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_RESIZE, rev, _e_border_event_border_resize_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_raise(E_Border *bd)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
|
|
e_container_border_raise(bd);
|
|
|
|
{
|
|
E_Event_Border_Raise *ev;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Raise));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ev->above = NULL;
|
|
ecore_event_add(E_EVENT_BORDER_RAISE, ev, _e_border_event_border_raise_free, NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_lower(E_Border *bd)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
|
|
e_container_border_lower(bd);
|
|
|
|
{
|
|
E_Event_Border_Lower *ev;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Lower));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ev->below = NULL;
|
|
ecore_event_add(E_EVENT_BORDER_LOWER, ev, _e_border_event_border_lower_free, NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_stack_above(E_Border *bd, E_Border *above)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
|
|
e_container_border_stack_above(bd, above);
|
|
|
|
{
|
|
E_Event_Border_Raise *ev;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Raise));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ev->above = above;
|
|
e_object_ref(E_OBJECT(above));
|
|
ecore_event_add(E_EVENT_BORDER_RAISE, ev, _e_border_event_border_raise_free, NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_stack_below(E_Border *bd, E_Border *below)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
|
|
e_container_border_stack_below(bd, below);
|
|
{
|
|
E_Event_Border_Lower *ev;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Lower));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ev->below = below;
|
|
e_object_ref(E_OBJECT(below));
|
|
ecore_event_add(E_EVENT_BORDER_LOWER, ev, _e_border_event_border_lower_free, NULL);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_focus_set(E_Border *bd, int focus, int set)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (!bd->client.icccm.accepts_focus) return;
|
|
// printf("flag focus to %i\n", focus);
|
|
if ((focus) && (!bd->focused))
|
|
edje_object_signal_emit(bd->bg_object, "active", "");
|
|
else if ((!focus) && (bd->focused))
|
|
edje_object_signal_emit(bd->bg_object, "passive", "");
|
|
bd->focused = focus;
|
|
if (set)
|
|
{
|
|
// printf("send focus to %i\n", focus);
|
|
if (bd->focused)
|
|
{
|
|
if ((focused != bd) && (focused))
|
|
e_border_focus_set(focused, 0, 0);
|
|
if (bd->client.icccm.take_focus)
|
|
{
|
|
// printf("take focus!\n");
|
|
/* this is a problem - basically we ASK the client to TAKE the focus itself
|
|
* BUT if a whole stream of events is happening, the client may take the focus
|
|
* LATER after we have gone and reset it back to somewhere else, thus it steals
|
|
* the focus away from where it should be (due to x being async etc.). no matter
|
|
* how nice and ICCCM this is - it's a major design flaw (imho) in ICCCM as it
|
|
* becomes nigh impossible for the wm then to re-serialise events and get the
|
|
* focus back to where it should be.
|
|
*
|
|
* example scenario of the bug:
|
|
*
|
|
* mouse enter window X
|
|
* wm set focus to X
|
|
* mouse leaves window X
|
|
* remove focus from window X
|
|
* mouse enters window Y
|
|
* wm asks client Y to "take the focus"
|
|
* mouse instantly moves back to window X
|
|
* wm sets focus on X
|
|
* suddenly focus is stolen by client Y as it finally recieved the request and took the focus
|
|
*
|
|
* now the focus is on Y where it should be on X
|
|
*/
|
|
ecore_x_window_focus(bd->client.win);
|
|
ecore_x_icccm_take_focus_send(bd->client.win, ecore_x_current_time_get());
|
|
// e_hints_active_window_set(bd->zone->container->manager, bd);
|
|
}
|
|
else
|
|
{
|
|
// printf("set focus\n");
|
|
ecore_x_window_focus(bd->client.win);
|
|
// e_hints_active_window_set(bd->zone->container->manager, bd);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// printf("remove focus\n");
|
|
ecore_x_window_focus(bd->zone->container->manager->root);
|
|
// e_hints_active_window_set(bd->zone->container->manager, NULL);
|
|
}
|
|
}
|
|
if ((bd->focused) && (focused != bd))
|
|
{
|
|
focused = bd;
|
|
e_hints_active_window_set(bd->zone->container->manager, bd);
|
|
}
|
|
else if ((!bd->focused) && (focused == bd))
|
|
{
|
|
focused = NULL;
|
|
e_hints_active_window_set(bd->zone->container->manager, NULL);
|
|
}
|
|
// printf("F %x %i\n", bd->client.win, bd->focused);
|
|
}
|
|
|
|
void
|
|
e_border_shade(E_Border *bd, E_Direction dir)
|
|
{
|
|
E_Event_Border_Resize *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (bd->maximized) return;
|
|
if (!bd->shaded)
|
|
{
|
|
// printf("SHADE!\n");
|
|
|
|
bd->shade.x = bd->x;
|
|
bd->shade.y = bd->y;
|
|
bd->shade.dir = dir;
|
|
|
|
e_hints_window_shaded_set(bd, 1);
|
|
e_hints_window_shade_direction_set(bd, dir);
|
|
|
|
if (e_config->border_shade_animate)
|
|
{
|
|
bd->shade.start = ecore_time_get();
|
|
bd->shading = 1;
|
|
bd->changes.shading = 1;
|
|
bd->changed = 1;
|
|
|
|
if (bd->shade.dir == E_DIRECTION_UP ||
|
|
bd->shade.dir == E_DIRECTION_LEFT)
|
|
ecore_x_window_gravity_set(bd->client.win, ECORE_X_GRAVITY_SW);
|
|
else
|
|
ecore_x_window_gravity_set(bd->client.win, ECORE_X_GRAVITY_NE);
|
|
|
|
bd->shade.anim = ecore_animator_add(_e_border_shade_animator, bd);
|
|
edje_object_signal_emit(bd->bg_object, "shading", "");
|
|
}
|
|
else
|
|
{
|
|
if (bd->shade.dir == E_DIRECTION_UP)
|
|
{
|
|
bd->h = bd->client_inset.t + bd->client_inset.b;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_DOWN)
|
|
{
|
|
bd->h = bd->client_inset.t + bd->client_inset.b;
|
|
bd->y = bd->y + bd->client.h;
|
|
bd->changes.pos = 1;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_LEFT)
|
|
{
|
|
bd->w = bd->client_inset.l + bd->client_inset.r;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_RIGHT)
|
|
{
|
|
bd->w = bd->client_inset.l + bd->client_inset.r;
|
|
bd->x = bd->x + bd->client.w;
|
|
bd->changes.pos = 1;
|
|
}
|
|
|
|
if ((bd->shaped) || (bd->client.shaped))
|
|
{
|
|
bd->need_shape_merge = 1;
|
|
bd->need_shape_export = 1;
|
|
}
|
|
bd->changes.size = 1;
|
|
bd->shaded = 1;
|
|
bd->changes.shaded = 1;
|
|
bd->changed = 1;
|
|
edje_object_signal_emit(bd->bg_object, "shaded", "");
|
|
ev = calloc(1, sizeof(E_Event_Border_Resize));
|
|
ev->border = bd;
|
|
/* The resize is added in the animator when animation complete */
|
|
/* For non-animated, we add it immediately with the new size */
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_RESIZE, ev, _e_border_event_border_resize_free, NULL);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_unshade(E_Border *bd, E_Direction dir)
|
|
{
|
|
E_Event_Border_Resize *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if (bd->maximized) return;
|
|
if (bd->shaded)
|
|
{
|
|
// printf("UNSHADE!\n");
|
|
|
|
bd->shade.dir = dir;
|
|
|
|
e_hints_window_shaded_set(bd, 0);
|
|
e_hints_window_shade_direction_set(bd, dir);
|
|
|
|
if (bd->shade.dir == E_DIRECTION_UP ||
|
|
bd->shade.dir == E_DIRECTION_LEFT)
|
|
{
|
|
bd->shade.x = bd->x;
|
|
bd->shade.y = bd->y;
|
|
}
|
|
else
|
|
{
|
|
bd->shade.x = bd->x - bd->client.w;
|
|
bd->shade.y = bd->y - bd->client.h;
|
|
}
|
|
if (e_config->border_shade_animate)
|
|
{
|
|
bd->shade.start = ecore_time_get();
|
|
bd->shading = 1;
|
|
bd->changes.shading = 1;
|
|
bd->changed = 1;
|
|
|
|
if (bd->shade.dir == E_DIRECTION_UP ||
|
|
bd->shade.dir == E_DIRECTION_LEFT)
|
|
ecore_x_window_gravity_set(bd->client.win, ECORE_X_GRAVITY_SW);
|
|
else
|
|
ecore_x_window_gravity_set(bd->client.win, ECORE_X_GRAVITY_NE);
|
|
|
|
bd->shade.anim = ecore_animator_add(_e_border_shade_animator, bd);
|
|
edje_object_signal_emit(bd->bg_object, "unshading", "");
|
|
}
|
|
else
|
|
{
|
|
if (bd->shade.dir == E_DIRECTION_UP)
|
|
{
|
|
bd->h = bd->client_inset.t + bd->client.h + bd->client_inset.b;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_DOWN)
|
|
{
|
|
bd->h = bd->client_inset.t + bd->client.h + bd->client_inset.b;
|
|
bd->y = bd->y - bd->client.h;
|
|
bd->changes.pos = 1;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_LEFT)
|
|
{
|
|
bd->w = bd->client_inset.l + bd->client.w + bd->client_inset.r;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_RIGHT)
|
|
{
|
|
bd->w = bd->client_inset.l + bd->client.w + bd->client_inset.r;
|
|
bd->x = bd->x - bd->client.w;
|
|
bd->changes.pos = 1;
|
|
}
|
|
if ((bd->shaped) || (bd->client.shaped))
|
|
{
|
|
bd->need_shape_merge = 1;
|
|
bd->need_shape_export = 1;
|
|
}
|
|
bd->changes.size = 1;
|
|
bd->shaded = 0;
|
|
bd->changes.shaded = 1;
|
|
bd->changed = 1;
|
|
edje_object_signal_emit(bd->bg_object, "unshaded", "");
|
|
ev = calloc(1, sizeof(E_Event_Border_Resize));
|
|
ev->border = bd;
|
|
/* The resize is added in the animator when animation complete */
|
|
/* For non-animated, we add it immediately with the new size */
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_RESIZE, ev, _e_border_event_border_resize_free, NULL);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_maximize(E_Border *bd)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
|
|
if ((bd->shaded) || (bd->shading)) return;
|
|
if (!bd->maximized)
|
|
{
|
|
// printf("MAXIMIZE!!\n");
|
|
bd->saved.x = bd->x;
|
|
bd->saved.y = bd->y;
|
|
bd->saved.w = bd->w;
|
|
bd->saved.h = bd->h;
|
|
|
|
e_hints_window_maximized_set(bd, 1);
|
|
|
|
/* FIXME maximize intelligently */
|
|
e_border_raise(bd);
|
|
e_border_move_resize(bd, bd->zone->x, bd->zone->y, bd->zone->w, bd->zone->h);
|
|
bd->maximized = 1;
|
|
bd->changes.pos = 1;
|
|
bd->changes.size = 1;
|
|
bd->changed = 1;
|
|
|
|
edje_object_signal_emit(bd->bg_object, "maximize", "");
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_unmaximize(E_Border *bd)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if ((bd->shaded) || (bd->shading)) return;
|
|
if (bd->maximized)
|
|
{
|
|
// printf("UNMAXIMIZE!!\n");
|
|
e_hints_window_maximized_set(bd, 0);
|
|
|
|
e_border_move_resize(bd, bd->saved.x, bd->saved.y, bd->saved.w, bd->saved.h);
|
|
|
|
bd->maximized = 0;
|
|
bd->changes.pos = 1;
|
|
bd->changes.size = 1;
|
|
bd->changed = 1;
|
|
|
|
edje_object_signal_emit(bd->bg_object, "unmaximize", "");
|
|
}
|
|
}
|
|
void
|
|
e_border_fullscreen(E_Border *bd)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
|
|
if ((bd->shaded) || (bd->shading)) return;
|
|
if (!bd->fullscreen)
|
|
{
|
|
int layer;
|
|
|
|
bd->saved.x = bd->x;
|
|
bd->saved.y = bd->y;
|
|
bd->saved.w = bd->w;
|
|
bd->saved.h = bd->h;
|
|
|
|
e_hints_window_fullscreen_set(bd, 1);
|
|
|
|
layer = bd->layer;
|
|
bd->layer = 200;
|
|
e_border_raise(bd);
|
|
bd->layer = layer;
|
|
e_border_move_resize(bd,
|
|
bd->zone->x - bd->client_inset.l,
|
|
bd->zone->y - bd->client_inset.t,
|
|
bd->zone->w + bd->client_inset.l + bd->client_inset.r,
|
|
bd->zone->h + bd->client_inset.t + bd->client_inset.b);
|
|
|
|
bd->fullscreen = 1;
|
|
bd->changes.pos = 1;
|
|
bd->changes.size = 1;
|
|
bd->changed = 1;
|
|
|
|
edje_object_signal_emit(bd->bg_object, "fullscreen", "");
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_unfullscreen(E_Border *bd)
|
|
{
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if ((bd->shaded) || (bd->shading)) return;
|
|
if (bd->fullscreen)
|
|
{
|
|
e_hints_window_fullscreen_set(bd, 0);
|
|
|
|
e_border_move_resize(bd, bd->saved.x, bd->saved.y, bd->saved.w, bd->saved.h);
|
|
|
|
bd->fullscreen = 0;
|
|
bd->changes.pos = 1;
|
|
bd->changes.size = 1;
|
|
bd->changed = 1;
|
|
|
|
e_border_raise(bd);
|
|
|
|
edje_object_signal_emit(bd->bg_object, "unfullscreen", "");
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_iconify(E_Border *bd)
|
|
{
|
|
E_Event_Border_Iconify *ev;
|
|
|
|
unsigned int iconic;
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if ((bd->shading)) return;
|
|
if (!bd->iconic)
|
|
{
|
|
bd->iconic = 1;
|
|
e_border_hide(bd, 1);
|
|
edje_object_signal_emit(bd->bg_object, "iconify", "");
|
|
}
|
|
iconic = 1;
|
|
e_hints_window_iconic_set(bd);
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_MAPPED, &iconic, 1);
|
|
|
|
ev = E_NEW(E_Event_Border_Iconify, 1);
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_ICONIFY, ev, _e_border_event_border_iconify_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_uniconify(E_Border *bd)
|
|
{
|
|
E_Desk *desk;
|
|
E_Event_Border_Uniconify *ev;
|
|
unsigned int iconic;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
if ((bd->shading)) return;
|
|
if (bd->iconic)
|
|
{
|
|
desk = e_desk_current_get(bd->desk->zone);
|
|
e_border_desk_set(bd, desk);
|
|
bd->iconic = 0;
|
|
e_border_show(bd);
|
|
e_border_raise(bd);
|
|
edje_object_signal_emit(bd->bg_object, "uniconify", "");
|
|
}
|
|
iconic = 0;
|
|
ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_MAPPED, &iconic, 1);
|
|
|
|
ev = E_NEW(E_Event_Border_Uniconify, 1);
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_UNICONIFY, ev, _e_border_event_border_uniconify_free, NULL);
|
|
|
|
}
|
|
|
|
void
|
|
e_border_stick(E_Border *bd)
|
|
{
|
|
E_Event_Border_Stick *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
bd->sticky = 1;
|
|
e_hints_window_sticky_set(bd, 1);
|
|
|
|
ev = E_NEW(E_Event_Border_Stick, 1);
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_STICK, ev, _e_border_event_border_stick_free, NULL);
|
|
}
|
|
|
|
void
|
|
e_border_unstick(E_Border *bd)
|
|
{
|
|
E_Event_Border_Unstick *ev;
|
|
|
|
E_OBJECT_CHECK(bd);
|
|
E_OBJECT_TYPE_CHECK(bd, E_BORDER_TYPE);
|
|
/* Set the desk before we unstick the border */
|
|
e_border_desk_set(bd, e_desk_current_get(bd->zone));
|
|
bd->sticky = 0;
|
|
e_hints_window_sticky_set(bd, 0);
|
|
|
|
ev = E_NEW(E_Event_Border_Unstick, 1);
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_UNSTICK, ev, _e_border_event_border_unstick_free, NULL);
|
|
}
|
|
|
|
E_Border *
|
|
e_border_find_by_client_window(Ecore_X_Window win)
|
|
{
|
|
Evas_List *l;
|
|
|
|
for (l = borders; l; l = l->next)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = l->data;
|
|
if (bd->client.win == win)
|
|
{
|
|
if (!e_object_is_del(E_OBJECT(bd)))
|
|
return bd;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
E_Border *
|
|
e_border_find_by_frame_window(Ecore_X_Window win)
|
|
{
|
|
Evas_List *l;
|
|
|
|
for (l = borders; l; l = l->next)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = l->data;
|
|
if (bd->bg_win == win)
|
|
{
|
|
if (!e_object_is_del(E_OBJECT(bd)))
|
|
return bd;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
E_Border *
|
|
e_border_find_by_window(Ecore_X_Window win)
|
|
{
|
|
Evas_List *l;
|
|
|
|
for (l = borders; l; l = l->next)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = l->data;
|
|
if (bd->win == win)
|
|
{
|
|
if (!e_object_is_del(E_OBJECT(bd)))
|
|
return bd;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
E_Border *
|
|
e_border_focused_get(void)
|
|
{
|
|
return focused;
|
|
}
|
|
|
|
void
|
|
e_border_idler_before(void)
|
|
{
|
|
Evas_List *ml, *cl;
|
|
|
|
if (!borders)
|
|
return;
|
|
|
|
/* We need to loop two times through the borders.
|
|
* 1. show windows
|
|
* 2. hide windows and evaluate rest
|
|
*/
|
|
for (ml = e_manager_list(); ml; ml = ml->next)
|
|
{
|
|
E_Manager *man;
|
|
|
|
man = ml->data;
|
|
for (cl = man->containers; cl; cl = cl->next)
|
|
{
|
|
E_Container *con;
|
|
E_Border_List *bl;
|
|
E_Border *bd;
|
|
|
|
con = cl->data;
|
|
bl = e_container_border_list_last(con);
|
|
while ((bd = e_container_border_list_prev(bl)))
|
|
{
|
|
if ((bd->changes.visible) && (bd->visible))
|
|
{
|
|
ecore_evas_show(bd->bg_ecore_evas);
|
|
ecore_x_window_show(bd->win);
|
|
bd->changes.visible = 0;
|
|
}
|
|
}
|
|
e_container_border_list_free(bl);
|
|
}
|
|
}
|
|
|
|
for (ml = e_manager_list(); ml; ml = ml->next)
|
|
{
|
|
E_Manager *man;
|
|
|
|
man = ml->data;
|
|
for (cl = man->containers; cl; cl = cl->next)
|
|
{
|
|
E_Container *con;
|
|
E_Border_List *bl;
|
|
E_Border *bd;
|
|
|
|
con = cl->data;
|
|
bl = e_container_border_list_first(con);
|
|
while ((bd = e_container_border_list_next(bl)))
|
|
{
|
|
if ((bd->changes.visible) && (!bd->visible))
|
|
{
|
|
ecore_x_window_hide(bd->win);
|
|
ecore_evas_hide(bd->bg_ecore_evas);
|
|
bd->changes.visible = 0;
|
|
}
|
|
if (bd->changed) _e_border_eval(bd);
|
|
}
|
|
e_container_border_list_free(bl);
|
|
}
|
|
}
|
|
}
|
|
|
|
Evas_List *
|
|
e_border_clients_get()
|
|
{
|
|
/* FIXME: This should be a somewhat ordered list */
|
|
return borders;
|
|
}
|
|
|
|
void
|
|
e_border_act_move_begin(E_Border *bd, Ecore_X_Event_Mouse_Button_Down *ev)
|
|
{
|
|
if (!bd->moving)
|
|
{
|
|
bd->moving = 1;
|
|
if (ev)
|
|
{
|
|
char source[256];
|
|
|
|
snprintf(source, sizeof(source) - 1, "mouse,%i", ev->button);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
}
|
|
e_border_raise(bd);
|
|
_e_border_move_begin(bd);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_act_move_end(E_Border *bd, Ecore_X_Event_Mouse_Button_Up *ev)
|
|
{
|
|
if (bd->moving)
|
|
{
|
|
bd->moving = 0;
|
|
_e_border_move_end(bd);
|
|
e_zone_flip_coords_handle(bd->zone, -1, -1);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_act_resize_begin(E_Border *bd, Ecore_X_Event_Mouse_Button_Down *ev)
|
|
{
|
|
if (bd->resize_mode == RESIZE_NONE)
|
|
{
|
|
if (bd->mouse.current.mx < (bd->x + bd-> w / 2))
|
|
{
|
|
if (bd->mouse.current.my < (bd->y + bd->h / 2))
|
|
{
|
|
bd->resize_mode = RESIZE_TL;
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_SE);
|
|
}
|
|
else
|
|
{
|
|
bd->resize_mode = RESIZE_BL;
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_NE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (bd->mouse.current.my < (bd->y + bd->h / 2))
|
|
{
|
|
bd->resize_mode = RESIZE_TR;
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_SW);
|
|
}
|
|
else
|
|
{
|
|
bd->resize_mode = RESIZE_BR;
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_NW);
|
|
}
|
|
}
|
|
if (ev)
|
|
{
|
|
char source[256];
|
|
|
|
snprintf(source, sizeof(source) - 1, "mouse,%i", ev->button);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
}
|
|
e_border_raise(bd);
|
|
_e_border_resize_begin(bd);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_act_resize_end(E_Border *bd, Ecore_X_Event_Mouse_Button_Up *ev)
|
|
{
|
|
if (bd->resize_mode != RESIZE_NONE)
|
|
{
|
|
bd->resize_mode = RESIZE_NONE;
|
|
_e_border_resize_end(bd);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_act_menu_begin(E_Border *bd, Ecore_X_Event_Mouse_Button_Down *ev, int key)
|
|
{
|
|
if (ev)
|
|
{
|
|
_e_border_menu_show(bd,
|
|
bd->x + ev->x - bd->zone->container->x,
|
|
bd->y + ev->y - bd->zone->container->y, key);
|
|
}
|
|
else
|
|
{
|
|
int x, y;
|
|
|
|
ecore_x_pointer_xy_get(bd->zone->container->win, &x, &y);
|
|
_e_border_menu_show(bd, x, y, key);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_act_close_begin(E_Border *bd)
|
|
{
|
|
if (bd->client.icccm.delete_request)
|
|
ecore_x_window_delete_request_send(bd->client.win);
|
|
else
|
|
{
|
|
ecore_x_kill(bd->client.win);
|
|
ecore_x_sync();
|
|
// ecore_x_window_del(bd->client.win);
|
|
e_border_hide(bd, 0);
|
|
e_object_del(E_OBJECT(bd));
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_act_kill_begin(E_Border *bd)
|
|
{
|
|
ecore_x_kill(bd->client.win);
|
|
ecore_x_sync();
|
|
e_border_hide(bd, 0);
|
|
e_object_del(E_OBJECT(bd));
|
|
}
|
|
|
|
void
|
|
e_border_button_bindings_ungrab_all(void)
|
|
{
|
|
Evas_List *l;
|
|
|
|
for (l = borders; l; l = l->next)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = l->data;
|
|
e_bindings_mouse_ungrab(E_BINDING_CONTEXT_BORDER, bd->win);
|
|
}
|
|
}
|
|
|
|
void
|
|
e_border_button_bindings_grab_all(void)
|
|
{
|
|
Evas_List *l;
|
|
|
|
for (l = borders; l; l = l->next)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = l->data;
|
|
e_bindings_mouse_grab(E_BINDING_CONTEXT_BORDER, bd->win);
|
|
}
|
|
}
|
|
|
|
/* local subsystem functions */
|
|
static void
|
|
_e_border_free(E_Border *bd)
|
|
{
|
|
printf("BD FREE %p\n", bd);
|
|
if (resize == bd)
|
|
_e_border_resize_end(bd);
|
|
if (move == bd)
|
|
_e_border_move_end(bd);
|
|
|
|
while (bd->pending_move_resize)
|
|
{
|
|
free(bd->pending_move_resize->data);
|
|
bd->pending_move_resize = evas_list_remove_list(bd->pending_move_resize, bd->pending_move_resize);
|
|
}
|
|
if (bd->border_menu)
|
|
{
|
|
e_object_del(E_OBJECT(bd->border_menu));
|
|
bd->border_menu = NULL;
|
|
}
|
|
if (focused == bd)
|
|
{
|
|
ecore_x_window_focus(bd->zone->container->manager->root);
|
|
focused = NULL;
|
|
}
|
|
while (bd->handlers)
|
|
{
|
|
Ecore_Event_Handler *h;
|
|
|
|
h = bd->handlers->data;
|
|
bd->handlers = evas_list_remove_list(bd->handlers, bd->handlers);
|
|
ecore_event_handler_del(h);
|
|
}
|
|
if (!bd->already_unparented)
|
|
{
|
|
ecore_x_window_reparent(bd->client.win, bd->zone->container->manager->root,
|
|
bd->x + bd->client_inset.l, bd->y + bd->client_inset.t);
|
|
ecore_x_window_save_set_del(bd->client.win);
|
|
}
|
|
if (bd->client.border.name) free(bd->client.border.name);
|
|
if (bd->client.icccm.title) free(bd->client.icccm.title);
|
|
if (bd->client.icccm.name) free(bd->client.icccm.name);
|
|
if (bd->client.icccm.class) free(bd->client.icccm.class);
|
|
if (bd->client.icccm.icon_name) free(bd->client.icccm.icon_name);
|
|
if (bd->client.icccm.machine) free(bd->client.icccm.machine);
|
|
e_object_del(E_OBJECT(bd->shape));
|
|
if (bd->icon_object) evas_object_del(bd->icon_object);
|
|
evas_object_del(bd->bg_object);
|
|
e_canvas_del(bd->bg_ecore_evas);
|
|
ecore_evas_free(bd->bg_ecore_evas);
|
|
ecore_x_window_del(bd->client.shell_win);
|
|
e_bindings_mouse_ungrab(E_BINDING_CONTEXT_BORDER, bd->win);
|
|
ecore_x_window_del(bd->win);
|
|
|
|
e_container_border_remove(bd);
|
|
borders = evas_list_remove(borders, bd);
|
|
|
|
free(bd);
|
|
}
|
|
|
|
static void
|
|
_e_border_del(E_Border *bd)
|
|
{
|
|
E_Event_Border_Remove *ev;
|
|
|
|
printf("BD DEL %p\n", bd);
|
|
ecore_x_window_reparent(bd->client.win,
|
|
bd->zone->container->manager->root,
|
|
bd->x + bd->client_inset.l,
|
|
bd->y + bd->client_inset.t);
|
|
ecore_x_window_save_set_del(bd->client.win);
|
|
bd->already_unparented = 1;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Remove));
|
|
ev->border = bd;
|
|
/* FIXME Don't ref this during shutdown. And the event is pointless
|
|
* during shutdown.. */
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_REMOVE, ev, _e_border_event_border_remove_free, NULL);
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_show_request(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Show_Request *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
e_border_show(bd);
|
|
e_border_raise(bd);
|
|
return 1;
|
|
}
|
|
|
|
static int _e_border_cb_window_destroy(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Destroy *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
e_border_hide(bd, 0);
|
|
e_object_del(E_OBJECT(bd));
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_hide(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Hide *e;
|
|
|
|
// printf("in hide cb\n");
|
|
bd = data;
|
|
e = ev;
|
|
printf("hide..\n");
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
printf("found %p\n");
|
|
if (bd->ignore_first_unmap > 0)
|
|
{
|
|
printf("IGNORE UNMAP\n");
|
|
bd->ignore_first_unmap--;
|
|
return 1;
|
|
}
|
|
/* Don't delete hidden or iconified windows */
|
|
if ((bd->iconic) || (!bd->visible))
|
|
{
|
|
printf("iconic %i || !visible %i\n",
|
|
bd->iconic, bd->visible);
|
|
e_border_hide(bd, 1);
|
|
}
|
|
else
|
|
{
|
|
printf("hide + del\n");
|
|
e_border_hide(bd, 0);
|
|
e_object_del(E_OBJECT(bd));
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_reparent(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Reparent *e;
|
|
|
|
bd = data;
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
// if (e->parent == bd->client.shell_win) return 1;
|
|
if (ecore_x_window_parent_get(e->win) == bd->client.shell_win) return 1;
|
|
e_border_hide(bd, 0);
|
|
e_object_del(E_OBJECT(bd));
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_configure_request(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Configure_Request *e;
|
|
|
|
bd = data;
|
|
e = ev;
|
|
printf("##- CONF REQ 0x%x , %iX%i+%i+%i\n",
|
|
e->win, e->w, e->h, e->x, e->y);
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd)
|
|
{
|
|
printf("generic config request 0x%x 0x%lx %i %i %ix%i %i 0x%x 0x%x...\n",
|
|
e->win, e->value_mask, e->x, e->y, e->w, e->h, e->border, e->abovewin, e->detail);
|
|
ecore_x_window_configure(e->win, e->value_mask,
|
|
e->x, e->y, e->w, e->h, e->border,
|
|
e->abovewin, e->detail);
|
|
return 1;
|
|
}
|
|
printf("##- CONFIGURE REQ 0x%0x mask: %c%c%c%c%c%c%c\n",
|
|
e->win,
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_X) ? 'X':' ',
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_Y) ? 'Y':' ',
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W) ? 'W':' ',
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H) ? 'H':' ',
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_BORDER_WIDTH) ? 'B':' ',
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_SIBLING) ? 'C':' ',
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_STACK_MODE) ? 'S':' '
|
|
);
|
|
|
|
if ((e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_X) ||
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_Y))
|
|
{
|
|
int x, y;
|
|
|
|
y = bd->y;
|
|
x = bd->x;
|
|
printf("##- ASK FOR 0x%x TO MOVE TO [FLG X%liY%li] %i,%i\n",
|
|
bd->client.win,
|
|
e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_X,
|
|
e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_Y,
|
|
x, y);
|
|
if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_X)
|
|
x = e->x;
|
|
if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_Y)
|
|
y = e->y;
|
|
if ((e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W) ||
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H))
|
|
{
|
|
int w, h;
|
|
|
|
h = bd->h;
|
|
w = bd->w;
|
|
if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W)
|
|
w = e->w + bd->client_inset.l + bd->client_inset.r;
|
|
if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H)
|
|
h = e->h + bd->client_inset.t + bd->client_inset.b;
|
|
printf("##- ASK FOR 0x%x TO RESIZE TO [FLG W%liH%li] %i,%i\n",
|
|
bd->client.win,
|
|
e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W,
|
|
e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H,
|
|
e->w, e->h);
|
|
e_border_move_resize(bd, x, y, w, h);
|
|
}
|
|
else
|
|
e_border_move(bd, x, y);
|
|
}
|
|
else if ((e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W) ||
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H))
|
|
{
|
|
int w, h;
|
|
|
|
h = bd->h;
|
|
w = bd->w;
|
|
if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W)
|
|
w = e->w + bd->client_inset.l + bd->client_inset.r;
|
|
if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H)
|
|
h = e->h + bd->client_inset.t + bd->client_inset.b;
|
|
printf("##- ASK FOR 0x%x TO RESIZE TO [FLG W%liH%li] %i,%i\n",
|
|
bd->client.win,
|
|
e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_W,
|
|
e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_H,
|
|
e->w, e->h);
|
|
e_border_resize(bd, w, h);
|
|
}
|
|
if ((e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_STACK_MODE) &&
|
|
(e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_SIBLING))
|
|
{
|
|
E_Border *obd;
|
|
|
|
if (e->detail == ECORE_X_WINDOW_STACK_ABOVE)
|
|
{
|
|
obd = e_border_find_by_client_window(e->abovewin);
|
|
if (obd)
|
|
e_border_stack_above(bd, obd);
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_BELOW)
|
|
{
|
|
obd = e_border_find_by_client_window(e->abovewin);
|
|
if (obd)
|
|
e_border_stack_below(bd, obd);
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_TOP_IF)
|
|
{
|
|
/* FIXME: do */
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_BOTTOM_IF)
|
|
{
|
|
/* FIXME: do */
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_OPPOSITE)
|
|
{
|
|
/* FIXME: do */
|
|
}
|
|
}
|
|
else if (e->value_mask & ECORE_X_WINDOW_CONFIGURE_MASK_STACK_MODE)
|
|
{
|
|
if (e->detail == ECORE_X_WINDOW_STACK_ABOVE)
|
|
{
|
|
e_border_raise(bd);
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_BELOW)
|
|
{
|
|
e_border_lower(bd);
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_TOP_IF)
|
|
{
|
|
/* FIXME: do */
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_BOTTOM_IF)
|
|
{
|
|
/* FIXME: do */
|
|
}
|
|
else if (e->detail == ECORE_X_WINDOW_STACK_OPPOSITE)
|
|
{
|
|
/* FIXME: do */
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_resize_request(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Resize_Request *e;
|
|
|
|
bd = data;
|
|
e = ev;
|
|
printf("##- RESZ REQ 0x%x , %iX%i\n",
|
|
e->win, e->w, e->h);
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd)
|
|
{
|
|
printf("generic resize request %x %ix%i ...\n",
|
|
e->win, e->w, e->h);
|
|
ecore_x_window_resize(e->win, e->w, e->h);
|
|
return 1;
|
|
}
|
|
printf("##- RESIZE REQ 0x%x\n", bd->client.win);
|
|
{
|
|
int w, h;
|
|
|
|
h = bd->h;
|
|
w = bd->w;
|
|
w = e->w + bd->client_inset.l + bd->client_inset.r;
|
|
h = e->h + bd->client_inset.t + bd->client_inset.b;
|
|
printf("##- ASK FOR 0x%x TO RESIZE TO %i,%i\n",
|
|
bd->client.win, e->w, e->h);
|
|
e_border_resize(bd, w, h);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_gravity(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Gravity *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
// printf("gravity for %0x\n", e->win);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_stack_request(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Stack_Request *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
// printf("stack req for %0x bd %p\n", e->win, bd);
|
|
if (!bd) return 1;
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_property(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Property *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
if (e->atom == ECORE_X_ATOM_WM_NAME)
|
|
{
|
|
bd->client.icccm.fetch.title = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_WM_CLASS)
|
|
{
|
|
bd->client.icccm.fetch.name_class = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_WM_ICON_NAME)
|
|
{
|
|
bd->client.icccm.fetch.icon_name = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_WM_CLIENT_MACHINE)
|
|
{
|
|
bd->client.icccm.fetch.machine = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_WM_PROTOCOLS)
|
|
{
|
|
bd->client.icccm.fetch.protocol = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_WM_HINTS)
|
|
{
|
|
bd->client.icccm.fetch.hints = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_WM_NORMAL_HINTS)
|
|
{
|
|
bd->client.icccm.fetch.size_pos_hints = 1;
|
|
bd->changed = 1;
|
|
}
|
|
else if (e->atom == ECORE_X_ATOM_MOTIF_WM_HINTS)
|
|
{
|
|
bd->client.mwm.fetch.hints = 1;
|
|
bd->changed = 1;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_colormap(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Colormap *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_shape(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Shape *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (bd)
|
|
{
|
|
bd->changes.shape = 1;
|
|
bd->changed = 1;
|
|
return 1;
|
|
}
|
|
bd = e_border_find_by_window(e->win);
|
|
if (bd)
|
|
{
|
|
bd->need_shape_export = 1;
|
|
bd->changed = 1;
|
|
return 1;
|
|
}
|
|
bd = e_border_find_by_frame_window(e->win);
|
|
if (bd)
|
|
{
|
|
bd->need_shape_merge = 1;
|
|
bd->changed = 1;
|
|
return 1;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_focus_in(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Focus_In *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
#ifdef INOUTDEBUG_FOCUS
|
|
{
|
|
time_t t;
|
|
char *ct;
|
|
|
|
const char *modes[] = {
|
|
"MODE_NORMAL",
|
|
"MODE_WHILE_GRABBED",
|
|
"MODE_GRAB",
|
|
"MODE_UNGRAB"
|
|
};
|
|
const char *details[] = {
|
|
"DETAIL_ANCESTOR",
|
|
"DETAIL_VIRTUAL",
|
|
"DETAIL_INFERIOR",
|
|
"DETAIL_NON_LINEAR",
|
|
"DETAIL_NON_LINEAR_VIRTUAL",
|
|
"DETAIL_POINTER",
|
|
"DETAIL_POINTER_ROOT",
|
|
"DETAIL_DETAIL_NONE"
|
|
};
|
|
t = time(NULL);
|
|
ct = ctime(&t);
|
|
ct[strlen(ct) - 1] = 0;
|
|
printf("FF ->IN %i 0x%x %s md=%s dt=%s\n",
|
|
e->time,
|
|
e->win,
|
|
ct,
|
|
modes[e->mode],
|
|
details[e->detail]);
|
|
}
|
|
#endif
|
|
e_border_focus_set(bd, 1, 0);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_focus_out(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_Focus_Out *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
#ifdef INOUTDEBUG_FOCUS
|
|
{
|
|
time_t t;
|
|
char *ct;
|
|
|
|
const char *modes[] = {
|
|
"MODE_NORMAL",
|
|
"MODE_WHILE_GRABBED",
|
|
"MODE_GRAB",
|
|
"MODE_UNGRAB"
|
|
};
|
|
const char *details[] = {
|
|
"DETAIL_ANCESTOR",
|
|
"DETAIL_VIRTUAL",
|
|
"DETAIL_INFERIOR",
|
|
"DETAIL_NON_LINEAR",
|
|
"DETAIL_NON_LINEAR_VIRTUAL",
|
|
"DETAIL_POINTER",
|
|
"DETAIL_POINTER_ROOT",
|
|
"DETAIL_DETAIL_NONE"
|
|
};
|
|
t = time(NULL);
|
|
ct = ctime(&t);
|
|
ct[strlen(ct) - 1] = 0;
|
|
printf("FF <-OUT %i 0x%x %s md=%s dt=%s\n",
|
|
e->time,
|
|
e->win,
|
|
ct,
|
|
modes[e->mode],
|
|
details[e->detail]);
|
|
}
|
|
#endif
|
|
if (e->mode == ECORE_X_EVENT_MODE_NORMAL)
|
|
{
|
|
if (e->detail == ECORE_X_EVENT_DETAIL_INFERIOR) return 1;
|
|
else if (e->detail == ECORE_X_EVENT_DETAIL_NON_LINEAR) return 1;
|
|
}
|
|
else if (e->mode == ECORE_X_EVENT_MODE_GRAB)
|
|
{
|
|
if (e->detail == ECORE_X_EVENT_DETAIL_NON_LINEAR) return 1;
|
|
else if (e->detail == ECORE_X_EVENT_DETAIL_INFERIOR) return 1;
|
|
else if (e->detail == ECORE_X_EVENT_DETAIL_NON_LINEAR_VIRTUAL) return 1;
|
|
else if (e->detail == ECORE_X_EVENT_DETAIL_ANCESTOR) return 1;
|
|
}
|
|
else if (e->mode == ECORE_X_EVENT_MODE_UNGRAB)
|
|
{
|
|
/* for firefox/thunderbird (xul) menu walking */
|
|
// if (e->detail == ECORE_X_EVENT_DETAIL_INFERIOR) return 1;
|
|
}
|
|
else if (e->mode == ECORE_X_EVENT_MODE_WHILE_GRABBED)
|
|
{
|
|
if (e->detail == ECORE_X_EVENT_DETAIL_ANCESTOR) return 1;
|
|
else if (e->detail == ECORE_X_EVENT_DETAIL_INFERIOR) return 1;
|
|
}
|
|
e_border_focus_set(bd, 0, 0);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_window_state(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Window_State *e;
|
|
int i, on;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (bd)
|
|
{
|
|
for (i = 0; i < 2; i++)
|
|
e_hints_window_state_update(bd, e->state[i], e->action);
|
|
}
|
|
else
|
|
{
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
switch (e->state[i])
|
|
{
|
|
case ECORE_X_WINDOW_STATE_ICONIFIED:
|
|
if (e->action == ECORE_X_WINDOW_STATE_ACTION_ADD)
|
|
{
|
|
ecore_x_icccm_state_set(e->win, ECORE_X_WINDOW_STATE_HINT_ICONIC);
|
|
ecore_x_netwm_window_state_set(e->win, ECORE_X_WINDOW_STATE_HIDDEN, 1);
|
|
}
|
|
break;
|
|
case ECORE_X_WINDOW_STATE_UNKNOWN:
|
|
case ECORE_X_WINDOW_STATE_HIDDEN:
|
|
/* Ignore */
|
|
break;
|
|
default:
|
|
switch (e->action)
|
|
{
|
|
case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
|
|
ecore_x_netwm_window_state_set(e->win, e->state[i], 0);
|
|
break;
|
|
case ECORE_X_WINDOW_STATE_ACTION_ADD:
|
|
ecore_x_netwm_window_state_set(e->win, e->state[i], 1);
|
|
break;
|
|
case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
|
|
on = ecore_x_netwm_window_state_isset(e->win, e->state[i]);
|
|
if (on)
|
|
ecore_x_netwm_window_state_set(e->win, e->state[i], 0);
|
|
else
|
|
ecore_x_netwm_window_state_set(e->win, e->state[i], 1);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_desktop_change(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Desktop_Change *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (bd)
|
|
{
|
|
if (e->desk == 0xffffffff)
|
|
e_border_stick(bd);
|
|
else if (e->desk < (bd->zone->desk_x_count * bd->zone->desk_y_count))
|
|
{
|
|
E_Desk *desk;
|
|
|
|
desk = e_desk_at_pos_get(bd->zone, e->desk);
|
|
if (desk)
|
|
e_border_desk_set(bd, desk);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ecore_x_netwm_desktop_set(e->win, e->desk);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_client_message(void *data, int ev_type, void *ev)
|
|
{
|
|
E_Border *bd;
|
|
Ecore_X_Event_Client_Message *e;
|
|
|
|
e = ev;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) return 1;
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_move_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
bd->moving = 1;
|
|
_e_border_moveinfo_gather(bd, source);
|
|
_e_border_move_begin(bd);
|
|
e_border_raise(bd);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_move_stop(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
bd->moving = 0;
|
|
_e_border_move_end(bd);
|
|
e_zone_flip_coords_handle(bd->zone, -1, -1);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_tl_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_TL;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_SE);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_t_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_T;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_S);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_tr_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_TR;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_SW);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_r_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_R;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_W);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_br_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_BR;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_NW);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_b_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_B;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_N);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_bl_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_BL;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_NE);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_l_start(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
bd->resize_mode = RESIZE_L;
|
|
_e_border_resize_begin(bd);
|
|
_e_border_moveinfo_gather(bd, source);
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_E);
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_resize_stop(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if ((bd->shaded) || (bd->shading) || (bd->maximized)) return;
|
|
|
|
_e_border_resize_handle(bd);
|
|
bd->resize_mode = RESIZE_NONE;
|
|
_e_border_resize_end(bd);
|
|
bd->changes.reset_gravity = 1;
|
|
bd->changed = 1;
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_action(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
|
|
if (e_dnd_active()) return;
|
|
|
|
if (!strcmp(source, "close"))
|
|
{
|
|
e_border_act_close_begin(bd);
|
|
}
|
|
else if (!strcmp(source, "shade_up") || !strcmp(source, "shade"))
|
|
{
|
|
if (bd->shaded) e_border_unshade(bd, E_DIRECTION_UP);
|
|
else e_border_shade(bd, E_DIRECTION_UP);
|
|
}
|
|
else if (!strcmp(source, "shade_down"))
|
|
{
|
|
if (bd->shaded) e_border_unshade(bd, E_DIRECTION_DOWN);
|
|
else e_border_shade(bd, E_DIRECTION_DOWN);
|
|
}
|
|
else if (!strcmp(source, "shade_left"))
|
|
{
|
|
if (bd->shaded) e_border_unshade(bd, E_DIRECTION_LEFT);
|
|
else e_border_shade(bd, E_DIRECTION_LEFT);
|
|
}
|
|
else if (!strcmp(source, "shade_right"))
|
|
{
|
|
if (bd->shaded) e_border_unshade(bd, E_DIRECTION_RIGHT);
|
|
else e_border_shade(bd, E_DIRECTION_RIGHT);
|
|
}
|
|
else if (!strcmp(source, "maximize"))
|
|
{
|
|
if (bd->maximized) e_border_unmaximize(bd);
|
|
else e_border_maximize(bd);
|
|
}
|
|
else if (!strcmp(source, "iconify"))
|
|
{
|
|
if (bd->iconic) e_border_uniconify(bd);
|
|
else e_border_iconify(bd);
|
|
}
|
|
else if (!strcmp(source, "menu"))
|
|
{
|
|
Evas_Coord x, y;
|
|
|
|
evas_pointer_canvas_xy_get(bd->bg_evas , &x, &y);
|
|
_e_border_menu_show(bd, x + bd->x, y + bd->y, 0);
|
|
}
|
|
else if (!strcmp(source, "lower"))
|
|
{
|
|
e_border_lower(bd);
|
|
}
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_signal_drag(void *data, Evas_Object *obj, const char *emission, const char *source)
|
|
{
|
|
E_Border *bd;
|
|
|
|
if (!strcmp(source, "icon"))
|
|
{
|
|
bd = data;
|
|
bd->drag.start = 1;
|
|
bd->drag.x = -1;
|
|
bd->drag.y = -1;
|
|
}
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_mouse_in(void *data, int type, void *event)
|
|
{
|
|
Ecore_X_Event_Mouse_In *ev;
|
|
E_Border *bd;
|
|
|
|
ev = event;
|
|
bd = data;
|
|
#ifdef INOUTDEBUG_MOUSE
|
|
{
|
|
time_t t;
|
|
char *ct;
|
|
|
|
const char *modes[] = {
|
|
"MODE_NORMAL",
|
|
"MODE_WHILE_GRABBED",
|
|
"MODE_GRAB",
|
|
"MODE_UNGRAB"
|
|
};
|
|
const char *details[] = {
|
|
"DETAIL_ANCESTOR",
|
|
"DETAIL_VIRTUAL",
|
|
"DETAIL_INFERIOR",
|
|
"DETAIL_NON_LINEAR",
|
|
"DETAIL_NON_LINEAR_VIRTUAL",
|
|
"DETAIL_POINTER",
|
|
"DETAIL_POINTER_ROOT",
|
|
"DETAIL_DETAIL_NONE"
|
|
};
|
|
t = time(NULL);
|
|
ct = ctime(&t);
|
|
ct[strlen(ct) - 1] = 0;
|
|
printf("@@ ->IN 0x%x 0x%x %s md=%s dt=%s\n",
|
|
ev->win, ev->event_win,
|
|
ct,
|
|
modes[ev->mode],
|
|
details[ev->detail]);
|
|
}
|
|
#endif
|
|
/*
|
|
if ((ev->mode == ECORE_X_EVENT_MODE_GRAB) &&
|
|
(ev->detail == ECORE_X_EVENT_DETAIL_NON_LINEAR_VIRTUAL) &&
|
|
(ev->win == bd->event_win) &&
|
|
(ev->event_win == bd->win))
|
|
return 1;
|
|
else if ((ev->mode == ECORE_X_EVENT_MODE_UNGRAB) &&
|
|
(ev->detail == ECORE_X_EVENT_DETAIL_NON_LINEAR_VIRTUAL) &&
|
|
(ev->win == bd->event_win) &&
|
|
(ev->event_win == bd->win))
|
|
return 1;
|
|
*/
|
|
if (ev->event_win == bd->win)
|
|
{
|
|
/* FIXME: this would normally put focus on the client on pointer */
|
|
/* focus - but click to focus it wouldnt */
|
|
e_border_focus_set(bd, 1, 1);
|
|
}
|
|
if (ev->win != bd->event_win) return 1;
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
evas_event_feed_mouse_move(bd->bg_evas, ev->x, ev->y, NULL);
|
|
evas_event_feed_mouse_in(bd->bg_evas, NULL);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_mouse_out(void *data, int type, void *event)
|
|
{
|
|
Ecore_X_Event_Mouse_Out *ev;
|
|
E_Border *bd;
|
|
|
|
ev = event;
|
|
bd = data;
|
|
#ifdef INOUTDEBUG_MOUSE
|
|
{
|
|
time_t t;
|
|
char *ct;
|
|
|
|
const char *modes[] = {
|
|
"MODE_NORMAL",
|
|
"MODE_WHILE_GRABBED",
|
|
"MODE_GRAB",
|
|
"MODE_UNGRAB"
|
|
};
|
|
const char *details[] = {
|
|
"DETAIL_ANCESTOR",
|
|
"DETAIL_VIRTUAL",
|
|
"DETAIL_INFERIOR",
|
|
"DETAIL_NON_LINEAR",
|
|
"DETAIL_NON_LINEAR_VIRTUAL",
|
|
"DETAIL_POINTER",
|
|
"DETAIL_POINTER_ROOT",
|
|
"DETAIL_DETAIL_NONE"
|
|
};
|
|
t = time(NULL);
|
|
ct = ctime(&t);
|
|
ct[strlen(ct) - 1] = 0;
|
|
printf("@@ <-OUT 0x%x 0x%x %s md=%s dt=%s\n",
|
|
ev->win, ev->event_win,
|
|
ct,
|
|
modes[ev->mode],
|
|
details[ev->detail]);
|
|
}
|
|
#endif
|
|
/* FIXME: this would normally take focus away in pointer focus mode */
|
|
// if (ev->mode == ECORE_X_EVENT_MODE_UNGRAB) return 1;
|
|
if (ev->event_win == bd->win)
|
|
{
|
|
if ((ev->mode == ECORE_X_EVENT_MODE_UNGRAB) &&
|
|
(ev->detail == ECORE_X_EVENT_DETAIL_INFERIOR))
|
|
return 1;
|
|
/* this is the out for pointer focus
|
|
if ((ev->mode == ECORE_X_EVENT_MODE_NORMAL) &&
|
|
(ev->detail == ECORE_X_EVENT_DETAIL_NON_LINEAR_VIRTUAL))
|
|
return 1;
|
|
*/
|
|
if (ev->mode == ECORE_X_EVENT_MODE_GRAB)
|
|
return 1;
|
|
if ((ev->mode == ECORE_X_EVENT_MODE_NORMAL) &&
|
|
(ev->detail == ECORE_X_EVENT_DETAIL_INFERIOR))
|
|
return 1;
|
|
e_border_focus_set(bd, 0, 1);
|
|
}
|
|
if (ev->win != bd->event_win) return 1;
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
evas_event_feed_mouse_move(bd->bg_evas, ev->x, ev->y, NULL);
|
|
evas_event_feed_mouse_out(bd->bg_evas, NULL);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_mouse_down(void *data, int type, void *event)
|
|
{
|
|
Ecore_X_Event_Mouse_Button_Down *ev;
|
|
E_Border *bd;
|
|
|
|
ev = event;
|
|
bd = data;
|
|
if (ev->event_win == bd->win)
|
|
{
|
|
if ((ev->button >= 1) && (ev->button <= 3))
|
|
{
|
|
bd->mouse.last_down[ev->button - 1].mx = ev->root.x;
|
|
bd->mouse.last_down[ev->button - 1].my = ev->root.y;
|
|
bd->mouse.last_down[ev->button - 1].x = bd->x;
|
|
bd->mouse.last_down[ev->button - 1].y = bd->y;
|
|
bd->mouse.last_down[ev->button - 1].w = bd->w;
|
|
bd->mouse.last_down[ev->button - 1].h = bd->h;
|
|
}
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
if (!bd->cur_mouse_action)
|
|
{
|
|
bd->cur_mouse_action =
|
|
e_bindings_mouse_down_event_handle(E_BINDING_CONTEXT_BORDER,
|
|
E_OBJECT(bd), ev);
|
|
}
|
|
}
|
|
if (ev->win != bd->event_win) return 1;
|
|
if ((ev->button >= 1) && (ev->button <= 3))
|
|
{
|
|
bd->mouse.last_down[ev->button - 1].mx = ev->root.x;
|
|
bd->mouse.last_down[ev->button - 1].my = ev->root.y;
|
|
bd->mouse.last_down[ev->button - 1].x = bd->x;
|
|
bd->mouse.last_down[ev->button - 1].y = bd->y;
|
|
bd->mouse.last_down[ev->button - 1].w = bd->w;
|
|
bd->mouse.last_down[ev->button - 1].h = bd->h;
|
|
}
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
if (bd->moving)
|
|
{
|
|
}
|
|
else if (bd->resize_mode != RESIZE_NONE)
|
|
{
|
|
}
|
|
else
|
|
{
|
|
Evas_Button_Flags flags = EVAS_BUTTON_NONE;
|
|
|
|
if (ev->double_click) flags |= EVAS_BUTTON_DOUBLE_CLICK;
|
|
if (ev->triple_click) flags |= EVAS_BUTTON_TRIPLE_CLICK;
|
|
evas_event_feed_mouse_move(bd->bg_evas, ev->x, ev->y, NULL);
|
|
evas_event_feed_mouse_down(bd->bg_evas, ev->button, flags, NULL);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_mouse_up(void *data, int type, void *event)
|
|
{
|
|
Ecore_X_Event_Mouse_Button_Up *ev;
|
|
E_Border *bd;
|
|
|
|
ev = event;
|
|
bd = data;
|
|
if (ev->event_win == bd->win)
|
|
{
|
|
if ((ev->button >= 1) && (ev->button <= 3))
|
|
{
|
|
bd->mouse.last_up[ev->button - 1].mx = ev->root.x;
|
|
bd->mouse.last_up[ev->button - 1].my = ev->root.y;
|
|
bd->mouse.last_up[ev->button - 1].x = bd->x;
|
|
bd->mouse.last_up[ev->button - 1].y = bd->y;
|
|
}
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
/* bug/problem. this action COULD be deleted during a move */
|
|
/* ... VERY unlikely though... VERY */
|
|
/* also we dont pass the same params that went in - then again that */
|
|
/* should be ok as we are just ending the action if it has an end */
|
|
if (bd->cur_mouse_action)
|
|
{
|
|
if (bd->cur_mouse_action->func.end_mouse)
|
|
bd->cur_mouse_action->func.end_mouse(E_OBJECT(bd), "", ev);
|
|
else if (bd->cur_mouse_action->func.end)
|
|
bd->cur_mouse_action->func.end(E_OBJECT(bd), "");
|
|
bd->cur_mouse_action = NULL;
|
|
}
|
|
else
|
|
e_bindings_mouse_up_event_handle(E_BINDING_CONTEXT_BORDER, E_OBJECT(bd), ev);
|
|
}
|
|
if (ev->win != bd->event_win) return 1;
|
|
if ((ev->button >= 1) && (ev->button <= 3))
|
|
{
|
|
bd->mouse.last_up[ev->button - 1].mx = ev->root.x;
|
|
bd->mouse.last_up[ev->button - 1].my = ev->root.y;
|
|
bd->mouse.last_up[ev->button - 1].x = bd->x;
|
|
bd->mouse.last_up[ev->button - 1].y = bd->y;
|
|
}
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
|
|
bd->drag.start = 0;
|
|
|
|
evas_event_feed_mouse_move(bd->bg_evas, ev->x, ev->y, NULL);
|
|
evas_event_feed_mouse_up(bd->bg_evas, ev->button, EVAS_BUTTON_NONE, NULL);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_mouse_move(void *data, int type, void *event)
|
|
{
|
|
Ecore_X_Event_Mouse_Move *ev;
|
|
E_Border *bd;
|
|
|
|
ev = event;
|
|
bd = data;
|
|
if (ev->event_win == bd->win)
|
|
{
|
|
// printf("GRABMOVE2!\n");
|
|
}
|
|
if ((ev->win != bd->event_win) &&
|
|
(ev->event_win != bd->win)) return 1;
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
if (bd->moving)
|
|
{
|
|
int x, y, new_x, new_y;
|
|
int new_w, new_h;
|
|
Evas_List *skiplist = NULL;
|
|
|
|
if ((bd->moveinfo.down.button >= 1) && (bd->moveinfo.down.button <= 3))
|
|
{
|
|
x = bd->mouse.last_down[bd->moveinfo.down.button - 1].x +
|
|
(bd->mouse.current.mx - bd->moveinfo.down.mx);
|
|
y = bd->mouse.last_down[bd->moveinfo.down.button - 1].y +
|
|
(bd->mouse.current.my - bd->moveinfo.down.my);
|
|
}
|
|
else
|
|
{
|
|
x = bd->x +
|
|
(bd->mouse.current.mx - bd->moveinfo.down.mx);
|
|
y = bd->y +
|
|
(bd->mouse.current.my - bd->moveinfo.down.my);
|
|
}
|
|
new_x = x;
|
|
new_y = y;
|
|
skiplist = evas_list_append(skiplist, bd);
|
|
e_resist_container_border_position(bd->zone->container, skiplist,
|
|
bd->x, bd->y, bd->w, bd->h,
|
|
x, y, bd->w, bd->h,
|
|
&new_x, &new_y, &new_w, &new_h);
|
|
evas_list_free(skiplist);
|
|
e_border_move(bd, new_x, new_y);
|
|
e_zone_flip_coords_handle(bd->zone, ev->root.x, ev->root.y);
|
|
}
|
|
else if (bd->resize_mode != RESIZE_NONE)
|
|
{
|
|
_e_border_resize_handle(bd);
|
|
}
|
|
else
|
|
{
|
|
if (bd->drag.start)
|
|
{
|
|
if ((bd->drag.x == -1) && (bd->drag.y == -1))
|
|
{
|
|
bd->drag.x = ev->root.x;
|
|
bd->drag.y = ev->root.y;
|
|
}
|
|
else
|
|
{
|
|
int x, y;
|
|
double dist;
|
|
|
|
x = bd->drag.x - ev->root.x;
|
|
y = bd->drag.y - ev->root.y;
|
|
dist = sqrt(pow(x, 2) + pow(y, 2));
|
|
if (dist > 4)
|
|
{
|
|
/* start drag! */
|
|
if (bd->icon_object)
|
|
{
|
|
E_Drag *drag;
|
|
Evas_Object *o;
|
|
Evas_Coord x, y, w, h;
|
|
const char *file, *part;
|
|
|
|
evas_object_geometry_get(bd->icon_object,
|
|
&x, &y, &w, &h);
|
|
drag = e_drag_new(bd->zone->container, bd->x + x, bd->y + y,
|
|
"enlightenment/border", bd, NULL);
|
|
o = edje_object_add(drag->evas);
|
|
edje_object_file_get(bd->icon_object, &file, &part);
|
|
edje_object_file_set(o, file, part);
|
|
e_drag_object_set(drag, o);
|
|
|
|
e_drag_resize(drag, w, h);
|
|
e_drag_start(drag, bd->drag.x, bd->drag.y);
|
|
evas_event_feed_mouse_up(bd->bg_evas, 1,
|
|
EVAS_BUTTON_NONE, NULL);
|
|
}
|
|
bd->drag.start = 0;
|
|
}
|
|
}
|
|
}
|
|
evas_event_feed_mouse_move(bd->bg_evas, ev->x, ev->y, NULL);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_mouse_wheel(void *data, int type, void *event)
|
|
{
|
|
Ecore_X_Event_Mouse_Wheel *ev;
|
|
E_Border *bd;
|
|
|
|
ev = event;
|
|
bd = data;
|
|
if (ev->win != bd->event_win) return 1;
|
|
bd->mouse.current.mx = ev->root.x;
|
|
bd->mouse.current.my = ev->root.y;
|
|
evas_event_feed_mouse_move(bd->bg_evas, ev->x, ev->y, NULL);
|
|
evas_event_feed_mouse_wheel(bd->bg_evas, ev->direction, ev->z, NULL);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_grab_replay(void *data, int type, void *event)
|
|
{
|
|
if (type == ECORE_X_EVENT_MOUSE_BUTTON_DOWN)
|
|
{
|
|
Ecore_X_Event_Mouse_Button_Down *e;
|
|
E_Border *bd;
|
|
|
|
e = event;
|
|
bd = e_border_find_by_client_window(e->win);
|
|
if (!bd) bd = e_border_find_by_client_window(e->event_win);
|
|
/* FIXME: return 1 if we pass this click on... */
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
_e_border_eval(E_Border *bd)
|
|
{
|
|
/* fetch any info queued to be fetched */
|
|
if (bd->client.icccm.fetch.title)
|
|
{
|
|
e_hints_window_name_get(bd);
|
|
bd->client.icccm.fetch.title = 0;
|
|
if (bd->bg_object)
|
|
{
|
|
edje_object_part_text_set(bd->bg_object, "title_text",
|
|
// "Japanese (hiragana): いろはにほへとちりぬるを");
|
|
bd->client.icccm.title);
|
|
// printf("SET TITLE %s\n", bd->client.icccm.title);
|
|
}
|
|
}
|
|
if (bd->client.icccm.fetch.name_class)
|
|
{
|
|
int nc_change = 0;
|
|
char *pname, *pclass;
|
|
|
|
pname = bd->client.icccm.name;
|
|
pclass = bd->client.icccm.class;
|
|
bd->client.icccm.name = NULL;
|
|
bd->client.icccm.class = NULL;
|
|
ecore_x_icccm_name_class_get(bd->client.win, &bd->client.icccm.name, &bd->client.icccm.class);
|
|
if ((pname) && (bd->client.icccm.name) &&
|
|
(pclass) && (bd->client.icccm.class))
|
|
{
|
|
if (!((!strcmp(bd->client.icccm.name, pname)) &&
|
|
(!strcmp(bd->client.icccm.class, pclass))))
|
|
nc_change = 1;
|
|
}
|
|
else if (((!pname) || (!pclass)) &&
|
|
((bd->client.icccm.name) || (bd->client.icccm.class)))
|
|
nc_change = 1;
|
|
else if (((bd->client.icccm.name) || (bd->client.icccm.class)) &&
|
|
((!pname) || (!pclass)))
|
|
nc_change = 1;
|
|
if (pname) free(pname);
|
|
if (pclass) free(pclass);
|
|
if (nc_change)
|
|
{
|
|
E_App *a;
|
|
|
|
a = NULL;
|
|
if (bd->icon_object)
|
|
{
|
|
evas_object_del(bd->icon_object);
|
|
bd->icon_object = NULL;
|
|
}
|
|
if ((bd->client.icccm.name) && (bd->client.icccm.class))
|
|
{
|
|
printf("name: %s, class: %s\n", bd->client.icccm.name, bd->client.icccm.class);
|
|
a = e_app_window_name_class_find(bd->client.icccm.name,
|
|
bd->client.icccm.class);
|
|
if (a)
|
|
{
|
|
bd->icon_object = edje_object_add(bd->bg_evas);
|
|
edje_object_file_set(bd->icon_object, a->path, "icon");
|
|
if (bd->bg_object)
|
|
{
|
|
evas_object_show(bd->icon_object);
|
|
edje_object_part_swallow(bd->bg_object, "icon_swallow", bd->icon_object);
|
|
}
|
|
else
|
|
{
|
|
evas_object_hide(bd->icon_object);
|
|
}
|
|
}
|
|
}
|
|
{
|
|
E_Event_Border_Icon_Change *ev;
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Icon_Change));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_ICON_CHANGE, ev, _e_border_event_border_icon_change_free, NULL);
|
|
}
|
|
}
|
|
bd->client.icccm.fetch.name_class = 0;
|
|
}
|
|
if (bd->client.icccm.fetch.icon_name)
|
|
{
|
|
if (bd->client.icccm.icon_name) free(bd->client.icccm.icon_name);
|
|
bd->client.icccm.icon_name = ecore_x_window_prop_icon_name_get(bd->client.win);
|
|
bd->client.icccm.fetch.icon_name = 0;
|
|
}
|
|
if (bd->client.icccm.fetch.machine)
|
|
{
|
|
if (bd->client.icccm.machine) free(bd->client.icccm.machine);
|
|
bd->client.icccm.machine = ecore_x_window_prop_client_machine_get(bd->client.win);
|
|
bd->client.icccm.fetch.machine = 0;
|
|
}
|
|
if (bd->client.icccm.fetch.hints)
|
|
{
|
|
int accepts_focus = 1;
|
|
int is_urgent = 0;
|
|
|
|
bd->client.icccm.initial_state = ECORE_X_WINDOW_STATE_HINT_NORMAL;
|
|
if (ecore_x_icccm_hints_get(bd->client.win,
|
|
&accepts_focus,
|
|
&bd->client.icccm.initial_state,
|
|
&bd->client.icccm.icon_pixmap,
|
|
&bd->client.icccm.icon_mask,
|
|
&bd->client.icccm.icon_window,
|
|
&bd->client.icccm.window_group,
|
|
&is_urgent))
|
|
{
|
|
bd->client.icccm.accepts_focus = accepts_focus;
|
|
bd->client.icccm.urgent = is_urgent;
|
|
|
|
/* If this is a new window, set the state as requested. */
|
|
if ((bd->new_client
|
|
&& (bd->client.icccm.initial_state == ECORE_X_WINDOW_STATE_HINT_ICONIC)))
|
|
e_border_iconify(bd);
|
|
}
|
|
bd->client.icccm.fetch.hints = 0;
|
|
}
|
|
if (bd->client.icccm.fetch.size_pos_hints)
|
|
{
|
|
int request_pos = 0;
|
|
|
|
if (ecore_x_icccm_size_pos_hints_get(bd->client.win,
|
|
&request_pos,
|
|
&bd->client.icccm.gravity,
|
|
&bd->client.icccm.min_w,
|
|
&bd->client.icccm.min_h,
|
|
&bd->client.icccm.max_w,
|
|
&bd->client.icccm.max_h,
|
|
&bd->client.icccm.base_w,
|
|
&bd->client.icccm.base_h,
|
|
&bd->client.icccm.step_w,
|
|
&bd->client.icccm.step_h,
|
|
&bd->client.icccm.min_aspect,
|
|
&bd->client.icccm.max_aspect))
|
|
{
|
|
bd->client.icccm.request_pos = request_pos;
|
|
}
|
|
else
|
|
{
|
|
printf("##- NO SIZE HINTS!\n");
|
|
}
|
|
if (bd->client.icccm.min_w > 32767) bd->client.icccm.min_w = 32767;
|
|
if (bd->client.icccm.min_h > 32767) bd->client.icccm.min_h = 32767;
|
|
if (bd->client.icccm.max_w > 32767) bd->client.icccm.max_w = 32767;
|
|
if (bd->client.icccm.max_h > 32767) bd->client.icccm.max_h = 32767;
|
|
if (bd->client.icccm.base_w > 32767) bd->client.icccm.base_w = 32767;
|
|
if (bd->client.icccm.base_h > 32767) bd->client.icccm.base_h = 32767;
|
|
if (bd->client.icccm.step_w < 1) bd->client.icccm.step_w = 1;
|
|
if (bd->client.icccm.step_h < 1) bd->client.icccm.step_h = 1;
|
|
printf("##- SIZE HINTS for 0x%x: min %ix%i, max %ix%i, base %ix%i, step %ix%i\n",
|
|
bd->client.win,
|
|
bd->client.icccm.min_w, bd->client.icccm.min_h,
|
|
bd->client.icccm.max_w, bd->client.icccm.max_h,
|
|
bd->client.icccm.base_w, bd->client.icccm.base_h,
|
|
bd->client.icccm.step_w, bd->client.icccm.step_h);
|
|
|
|
bd->client.icccm.fetch.size_pos_hints = 0;
|
|
}
|
|
if (bd->client.icccm.fetch.protocol)
|
|
{
|
|
int i, num;
|
|
Ecore_X_WM_Protocol *proto;
|
|
|
|
proto = ecore_x_window_prop_protocol_list_get(bd->client.win, &num);
|
|
if (proto)
|
|
{
|
|
for (i = 0; i < num; i++)
|
|
{
|
|
if (proto[i] == ECORE_X_WM_PROTOCOL_DELETE_REQUEST)
|
|
bd->client.icccm.delete_request = 1;
|
|
if (proto[i] == ECORE_X_WM_PROTOCOL_TAKE_FOCUS)
|
|
bd->client.icccm.take_focus = 1;
|
|
}
|
|
free(proto);
|
|
}
|
|
bd->client.icccm.fetch.protocol = 0;
|
|
}
|
|
if (bd->client.mwm.fetch.hints)
|
|
{
|
|
int pb;
|
|
|
|
bd->client.mwm.exists =
|
|
ecore_x_mwm_hints_get(bd->client.win,
|
|
&bd->client.mwm.func,
|
|
&bd->client.mwm.decor,
|
|
&bd->client.mwm.input);
|
|
pb = bd->client.mwm.borderless;
|
|
bd->client.mwm.borderless = 0;
|
|
if (bd->client.mwm.exists)
|
|
{
|
|
printf("##- MWM HINTS SET 0x%x!\n", bd->client.win);
|
|
if ((!(bd->client.mwm.decor & ECORE_X_MWM_HINT_DECOR_ALL)) &&
|
|
(!(bd->client.mwm.decor & ECORE_X_MWM_HINT_DECOR_TITLE)) &&
|
|
(!(bd->client.mwm.decor & ECORE_X_MWM_HINT_DECOR_BORDER)))
|
|
bd->client.mwm.borderless = 1;
|
|
}
|
|
if (bd->client.mwm.borderless != pb)
|
|
{
|
|
if (bd->client.border.name) free(bd->client.border.name);
|
|
if (bd->client.mwm.borderless)
|
|
bd->client.border.name = strdup("borderless");
|
|
else
|
|
bd->client.border.name = strdup("default");
|
|
if (bd->client.mwm.borderless)
|
|
printf("client %s borderless\n", bd->client.icccm.title);
|
|
bd->client.border.changed = 1;
|
|
}
|
|
bd->client.mwm.fetch.hints = 0;
|
|
}
|
|
|
|
if (bd->changes.shape)
|
|
{
|
|
Ecore_X_Rectangle *rects;
|
|
int num;
|
|
|
|
rects = ecore_x_window_shape_rectangles_get(bd->client.win, &num);
|
|
if (rects)
|
|
{
|
|
if ((num == 1) &&
|
|
(rects[0].x == 0) &&
|
|
(rects[0].y == 0) &&
|
|
(rects[0].width == bd->client.w) &&
|
|
(rects[0].height == bd->client.h))
|
|
{
|
|
if (bd->client.shaped)
|
|
{
|
|
bd->client.shaped = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!bd->client.shaped)
|
|
{
|
|
bd->client.shaped = 1;
|
|
}
|
|
}
|
|
free(rects);
|
|
}
|
|
bd->need_shape_merge = 1;
|
|
/* is the client shaped? */
|
|
bd->changes.shape = 0;
|
|
}
|
|
|
|
if (bd->client.border.changed)
|
|
{
|
|
Evas_Object *o;
|
|
char buf[4096];
|
|
Evas_Coord cx, cy, cw, ch;
|
|
int l, r, t, b;
|
|
int ok;
|
|
|
|
if (!bd->client.border.name)
|
|
{
|
|
bd->client.border.name = strdup("default");
|
|
}
|
|
if (bd->bg_object)
|
|
{
|
|
bd->w -= (bd->client_inset.l + bd->client_inset.r);
|
|
bd->h -= (bd->client_inset.t + bd->client_inset.b);
|
|
bd->client_inset.l = 0;
|
|
bd->client_inset.r = 0;
|
|
bd->client_inset.t = 0;
|
|
bd->client_inset.b = 0;
|
|
bd->changes.size = 1;
|
|
evas_object_del(bd->bg_object);
|
|
}
|
|
o = edje_object_add(bd->bg_evas);
|
|
snprintf(buf, sizeof(buf), "widgets/border/%s/border",
|
|
bd->client.border.name);
|
|
ok = e_theme_edje_object_set(o, "base/theme/borders", buf);
|
|
if (ok)
|
|
{
|
|
const char *shape_option;
|
|
|
|
bd->bg_object = o;
|
|
shape_option = edje_object_data_get(o, "shaped");
|
|
if (shape_option)
|
|
{
|
|
if (!strcmp(shape_option, "1"))
|
|
{
|
|
if (!bd->shaped)
|
|
{
|
|
bd->shaped = 1;
|
|
ecore_evas_shaped_set(bd->bg_ecore_evas, bd->shaped);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (bd->shaped)
|
|
{
|
|
bd->shaped = 0;
|
|
ecore_evas_shaped_set(bd->bg_ecore_evas, bd->shaped);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (bd->shaped)
|
|
{
|
|
bd->shaped = 0;
|
|
ecore_evas_shaped_set(bd->bg_ecore_evas, bd->shaped);
|
|
}
|
|
}
|
|
|
|
edje_object_part_text_set(o, "title_text",
|
|
// "Japanese (hiragana): いろはにほへとちりぬるを");
|
|
bd->client.icccm.title);
|
|
// printf("SET TITLE2 %s\n", bd->client.icccm.title);
|
|
evas_object_resize(o, 1000, 1000);
|
|
edje_object_calc_force(o);
|
|
edje_object_part_geometry_get(o, "client", &cx, &cy, &cw, &ch);
|
|
l = cx;
|
|
r = 1000 - (cx + cw);
|
|
t = cy;
|
|
b = 1000 - (cy + ch);
|
|
}
|
|
else
|
|
{
|
|
evas_object_del(o);
|
|
bd->bg_object = NULL;
|
|
l = 0;
|
|
r = 0;
|
|
t = 0;
|
|
b = 0;
|
|
}
|
|
bd->client_inset.l = l;
|
|
bd->client_inset.r = r;
|
|
bd->client_inset.t = t;
|
|
bd->client_inset.b = b;
|
|
ecore_x_netwm_frame_size_set(bd->client.win, l, r, t, b);
|
|
ecore_x_e_frame_size_set(bd->client.win, l, r, t, b);
|
|
bd->w += (bd->client_inset.l + bd->client_inset.r);
|
|
bd->h += (bd->client_inset.t + bd->client_inset.b);
|
|
bd->changes.size = 1;
|
|
ecore_x_window_move(bd->client.shell_win, l, t);
|
|
if (bd->bg_object)
|
|
{
|
|
edje_object_signal_callback_add(bd->bg_object, "move_start", "*",
|
|
_e_border_cb_signal_move_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "move_stop", "*",
|
|
_e_border_cb_signal_move_stop, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_tl_start", "*",
|
|
_e_border_cb_signal_resize_tl_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_t_start", "*",
|
|
_e_border_cb_signal_resize_t_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_tr_start", "*",
|
|
_e_border_cb_signal_resize_tr_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_r_start", "*",
|
|
_e_border_cb_signal_resize_r_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_br_start", "*",
|
|
_e_border_cb_signal_resize_br_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_b_start", "*",
|
|
_e_border_cb_signal_resize_b_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_bl_start", "*",
|
|
_e_border_cb_signal_resize_bl_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_l_start", "*",
|
|
_e_border_cb_signal_resize_l_start, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "resize_stop", "*",
|
|
_e_border_cb_signal_resize_stop, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "action", "*",
|
|
_e_border_cb_signal_action, bd);
|
|
edje_object_signal_callback_add(bd->bg_object, "drag", "*",
|
|
_e_border_cb_signal_drag, bd);
|
|
if (bd->focused)
|
|
edje_object_signal_emit(bd->bg_object, "active", "");
|
|
evas_object_move(bd->bg_object, 0, 0);
|
|
evas_object_resize(bd->bg_object, bd->w, bd->h);
|
|
evas_object_show(bd->bg_object);
|
|
}
|
|
bd->client.border.changed = 0;
|
|
|
|
if (bd->icon_object)
|
|
{
|
|
if (bd->bg_object)
|
|
{
|
|
evas_object_show(bd->icon_object);
|
|
edje_object_part_swallow(bd->bg_object, "icon_swallow", bd->icon_object);
|
|
}
|
|
else
|
|
{
|
|
evas_object_hide(bd->icon_object);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bd->new_client)
|
|
{
|
|
E_Event_Border_Add *ev;
|
|
|
|
bd->new_client = 0;
|
|
printf("##- NEW CLIENT SETUP 0x%x\n", bd->client.win);
|
|
if (bd->re_manage)
|
|
{
|
|
printf("##- REMANAGE!\n");
|
|
bd->x -= bd->client_inset.l;
|
|
bd->y -= bd->client_inset.t;
|
|
bd->changes.pos = 1;
|
|
}
|
|
else
|
|
{
|
|
if (bd->client.icccm.request_pos)
|
|
{
|
|
Ecore_X_Window_Attributes *att;
|
|
int bw;
|
|
|
|
att = &bd->client.initial_attributes;
|
|
printf("##- REQUEST POS 0x%x [%i,%i]\n",
|
|
bd->client.win, att->x, att->y);
|
|
bw = att->border * 2;
|
|
switch (bd->client.icccm.gravity)
|
|
{
|
|
case ECORE_X_GRAVITY_N:
|
|
bd->x = (att->x - (bw / 2));
|
|
bd->y = att->y;
|
|
break;
|
|
case ECORE_X_GRAVITY_NE:
|
|
bd->x = (att->x - (bw)) - (bd->client_inset.l);
|
|
bd->y = att->y;
|
|
break;
|
|
case ECORE_X_GRAVITY_E:
|
|
bd->x = (att->x - (bw)) - (bd->client_inset.l);
|
|
bd->y = (att->y - (bw / 2));
|
|
break;
|
|
case ECORE_X_GRAVITY_SE:
|
|
bd->x = (att->x - (bw)) - (bd->client_inset.l);
|
|
bd->y = (att->y - (bw)) - (bd->client_inset.t);
|
|
break;
|
|
case ECORE_X_GRAVITY_S:
|
|
bd->x = (att->x - (bw / 2));
|
|
bd->y = (att->y - (bw)) - (bd->client_inset.t);
|
|
break;
|
|
case ECORE_X_GRAVITY_SW:
|
|
bd->x = att->x;
|
|
bd->y = (att->y - (bw)) - (bd->client_inset.t);
|
|
break;
|
|
case ECORE_X_GRAVITY_W:
|
|
bd->x = att->x;
|
|
bd->y = (att->y - (bw)) - (bd->client_inset.t);
|
|
break;
|
|
case ECORE_X_GRAVITY_NW:
|
|
default:
|
|
bd->x = att->x;
|
|
bd->y = att->y;
|
|
}
|
|
bd->changes.pos = 1;
|
|
}
|
|
else
|
|
{
|
|
Evas_List *skiplist = NULL;
|
|
int new_x, new_y;
|
|
|
|
printf("##- AUTO POS 0x%x\n", bd->client.win);
|
|
if (bd->zone->w > bd->w)
|
|
new_x = bd->zone->x + (rand() % (bd->zone->w - bd->w));
|
|
else
|
|
new_x = bd->zone->x;
|
|
if (bd->zone->h > bd->h)
|
|
new_y = bd->zone->y + (rand() % (bd->zone->h - bd->h));
|
|
else
|
|
new_y = bd->zone->y;
|
|
|
|
skiplist = evas_list_append(skiplist, bd);
|
|
e_place_zone_region_smart(bd->zone, skiplist,
|
|
bd->x, bd->y, bd->w, bd->h,
|
|
&new_x, &new_y);
|
|
evas_list_free(skiplist);
|
|
bd->x = new_x;
|
|
bd->y = new_y;
|
|
bd->changes.pos = 1;
|
|
}
|
|
}
|
|
while (bd->pending_move_resize)
|
|
{
|
|
E_Border_Pending_Move_Resize *pnd;
|
|
|
|
pnd = bd->pending_move_resize->data;
|
|
if (pnd->move)
|
|
{
|
|
bd->x = pnd->x;
|
|
bd->y = pnd->y;
|
|
bd->changes.pos = 1;
|
|
}
|
|
if (pnd->resize)
|
|
{
|
|
bd->w = pnd->w + bd->client_inset.l + bd->client_inset.r;
|
|
bd->h = pnd->h + bd->client_inset.t + bd->client_inset.b;
|
|
bd->client.w = pnd->w;
|
|
bd->client.h = pnd->h;
|
|
bd->changes.size = 1;
|
|
}
|
|
free(pnd);
|
|
bd->pending_move_resize = evas_list_remove_list(bd->pending_move_resize, bd->pending_move_resize);
|
|
}
|
|
|
|
ev = calloc(1, sizeof(E_Event_Border_Add));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_ADD, ev, _e_border_event_border_add_free, NULL);
|
|
|
|
/* Recreate state */
|
|
e_hints_window_init(bd);
|
|
|
|
ecore_x_icccm_move_resize_send(bd->client.win,
|
|
bd->x + bd->client_inset.l,
|
|
bd->y + bd->client_inset.t,
|
|
bd->client.w,
|
|
bd->client.h);
|
|
}
|
|
|
|
/* effect changes to the window border itself */
|
|
if ((bd->changes.shading))
|
|
{
|
|
/* show at start of unshade (but don't hide until end of shade) */
|
|
if (bd->shaded)
|
|
ecore_x_window_raise(bd->client.shell_win);
|
|
bd->changes.shading = 0;
|
|
}
|
|
if ((bd->changes.shaded) && (bd->changes.pos) && (bd->changes.size))
|
|
{
|
|
if (bd->shaded)
|
|
ecore_x_window_lower(bd->client.shell_win);
|
|
else
|
|
ecore_x_window_raise(bd->client.shell_win);
|
|
bd->changes.shaded = 0;
|
|
}
|
|
else if ((bd->changes.shaded) && (bd->changes.pos))
|
|
{
|
|
if (bd->shaded)
|
|
ecore_x_window_lower(bd->client.shell_win);
|
|
else
|
|
ecore_x_window_raise(bd->client.shell_win);
|
|
bd->changes.size = 1;
|
|
bd->changes.shaded = 0;
|
|
}
|
|
else if ((bd->changes.shaded) && (bd->changes.size))
|
|
{
|
|
if (bd->shaded)
|
|
ecore_x_window_lower(bd->client.shell_win);
|
|
else
|
|
ecore_x_window_raise(bd->client.shell_win);
|
|
bd->changes.shaded = 0;
|
|
}
|
|
else if (bd->changes.shaded)
|
|
{
|
|
if (bd->shaded)
|
|
ecore_x_window_lower(bd->client.shell_win);
|
|
else
|
|
ecore_x_window_raise(bd->client.shell_win);
|
|
bd->changes.size = 1;
|
|
bd->changes.shaded = 0;
|
|
}
|
|
|
|
if ((bd->changes.pos) && (bd->changes.size))
|
|
{
|
|
printf("##- BORDER NEEDS POS/SIZE CHANGE 0x%x\n", bd->client.win);
|
|
if (bd->shaded && !bd->shading)
|
|
{
|
|
evas_obscured_clear(bd->bg_evas);
|
|
ecore_x_window_move_resize(bd->win, bd->x, bd->y, bd->w, bd->h);
|
|
ecore_x_window_move_resize(bd->event_win, 0, 0, bd->w, bd->h);
|
|
ecore_evas_move_resize(bd->bg_ecore_evas, 0, 0, bd->w, bd->h);
|
|
evas_object_resize(bd->bg_object, bd->w, bd->h);
|
|
e_container_shape_resize(bd->shape, bd->w, bd->h);
|
|
e_container_shape_move(bd->shape, bd->x, bd->y);
|
|
}
|
|
else
|
|
{
|
|
evas_obscured_clear(bd->bg_evas);
|
|
evas_obscured_rectangle_add(bd->bg_evas,
|
|
bd->client_inset.l, bd->client_inset.t,
|
|
bd->w - (bd->client_inset.l + bd->client_inset.r),
|
|
bd->h - (bd->client_inset.t + bd->client_inset.b));
|
|
ecore_x_window_move_resize(bd->win, bd->x, bd->y, bd->w, bd->h);
|
|
ecore_x_window_move_resize(bd->event_win, 0, 0, bd->w, bd->h);
|
|
ecore_x_window_move_resize(bd->client.shell_win,
|
|
bd->client_inset.l, bd->client_inset.t,
|
|
bd->w - (bd->client_inset.l + bd->client_inset.r),
|
|
bd->h - (bd->client_inset.t + bd->client_inset.b));
|
|
if (bd->shading)
|
|
{
|
|
if (bd->shade.dir == E_DIRECTION_UP)
|
|
ecore_x_window_move_resize(bd->client.win, 0,
|
|
bd->h - (bd->client_inset.t + bd->client_inset.b) -
|
|
bd->client.h,
|
|
bd->client.w, bd->client.h);
|
|
else if (bd->shade.dir == E_DIRECTION_LEFT)
|
|
ecore_x_window_move_resize(bd->client.win,
|
|
bd->w - (bd->client_inset.l + bd->client_inset.r) -
|
|
bd->client.h,
|
|
0, bd->client.w, bd->client.h);
|
|
else
|
|
ecore_x_window_move_resize(bd->client.win, 0, 0,
|
|
bd->client.w, bd->client.h);
|
|
}
|
|
else
|
|
ecore_x_window_move_resize(bd->client.win, 0, 0,
|
|
bd->client.w, bd->client.h);
|
|
ecore_evas_move_resize(bd->bg_ecore_evas, 0, 0, bd->w, bd->h);
|
|
evas_object_resize(bd->bg_object, bd->w, bd->h);
|
|
e_container_shape_resize(bd->shape, bd->w, bd->h);
|
|
e_container_shape_move(bd->shape, bd->x, bd->y);
|
|
}
|
|
bd->changes.pos = 0;
|
|
bd->changes.size = 0;
|
|
}
|
|
else if (bd->changes.pos)
|
|
{
|
|
ecore_x_window_move(bd->win, bd->x, bd->y);
|
|
e_container_shape_move(bd->shape, bd->x, bd->y);
|
|
bd->changes.pos = 0;
|
|
}
|
|
else if (bd->changes.size)
|
|
{
|
|
printf("##- BORDER NEEDS SIZE CHANGE 0x%x\n", bd->client.win);
|
|
if (bd->shaded && !bd->shading)
|
|
{
|
|
evas_obscured_clear(bd->bg_evas);
|
|
ecore_x_window_move_resize(bd->event_win, 0, 0, bd->w, bd->h);
|
|
ecore_x_window_resize(bd->win, bd->w, bd->h);
|
|
ecore_evas_move_resize(bd->bg_ecore_evas, 0, 0, bd->w, bd->h);
|
|
evas_object_resize(bd->bg_object, bd->w, bd->h);
|
|
e_container_shape_resize(bd->shape, bd->w, bd->h);
|
|
}
|
|
else
|
|
{
|
|
evas_obscured_clear(bd->bg_evas);
|
|
evas_obscured_rectangle_add(bd->bg_evas,
|
|
bd->client_inset.l, bd->client_inset.t,
|
|
bd->w - (bd->client_inset.l + bd->client_inset.r), bd->h - (bd->client_inset.t + bd->client_inset.b));
|
|
ecore_x_window_move_resize(bd->event_win, 0, 0, bd->w, bd->h);
|
|
ecore_x_window_resize(bd->win, bd->w, bd->h);
|
|
ecore_x_window_move_resize(bd->client.shell_win,
|
|
bd->client_inset.l, bd->client_inset.t,
|
|
bd->w - (bd->client_inset.l + bd->client_inset.r),
|
|
bd->h - (bd->client_inset.t + bd->client_inset.b));
|
|
if (bd->shading)
|
|
{
|
|
if (bd->shade.dir == E_DIRECTION_UP)
|
|
ecore_x_window_move_resize(bd->client.win, 0,
|
|
bd->h - (bd->client_inset.t + bd->client_inset.b) -
|
|
bd->client.h,
|
|
bd->client.w, bd->client.h);
|
|
else if (bd->shade.dir == E_DIRECTION_LEFT)
|
|
ecore_x_window_move_resize(bd->client.win,
|
|
bd->w - (bd->client_inset.l + bd->client_inset.r) -
|
|
bd->client.h,
|
|
0, bd->client.w, bd->client.h);
|
|
else
|
|
ecore_x_window_move_resize(bd->client.win, 0, 0,
|
|
bd->client.w, bd->client.h);
|
|
}
|
|
else
|
|
ecore_x_window_move_resize(bd->client.win, 0, 0,
|
|
bd->client.w, bd->client.h);
|
|
ecore_evas_move_resize(bd->bg_ecore_evas, 0, 0, bd->w, bd->h);
|
|
evas_object_resize(bd->bg_object, bd->w, bd->h);
|
|
e_container_shape_resize(bd->shape, bd->w, bd->h);
|
|
}
|
|
bd->changes.size = 0;
|
|
}
|
|
|
|
if (bd->changes.reset_gravity)
|
|
{
|
|
GRAV_SET(bd, ECORE_X_GRAVITY_NW);
|
|
bd->changes.reset_gravity = 0;
|
|
}
|
|
|
|
if (bd->need_shape_merge)
|
|
{
|
|
if ((bd->shaped) || (bd->client.shaped))
|
|
{
|
|
Ecore_X_Window twin, twin2;
|
|
int x, y;
|
|
|
|
twin = ecore_x_window_override_new(bd->win, 0, 0, bd->w, bd->h);
|
|
if (bd->shaped)
|
|
{
|
|
ecore_x_window_shape_window_set(twin, bd->bg_win);
|
|
}
|
|
else
|
|
{
|
|
Ecore_X_Rectangle rects[4];
|
|
|
|
rects[0].x = 0;
|
|
rects[0].y = 0;
|
|
rects[0].width = bd->w;
|
|
rects[0].height = bd->client_inset.t;
|
|
rects[1].x = 0;
|
|
rects[1].y = bd->client_inset.t;
|
|
rects[1].width = bd->client_inset.l;
|
|
rects[1].height = bd->h - bd->client_inset.t - bd->client_inset.b;
|
|
rects[2].x = bd->w - bd->client_inset.r;
|
|
rects[2].y = bd->client_inset.t;
|
|
rects[2].width = bd->client_inset.r;
|
|
rects[2].height = bd->h - bd->client_inset.t - bd->client_inset.b;
|
|
rects[3].x = 0;
|
|
rects[3].y = bd->h - bd->client_inset.b;
|
|
rects[3].width = bd->w;
|
|
rects[3].height = bd->client_inset.b;
|
|
ecore_x_window_shape_rectangles_set(twin, rects, 4);
|
|
}
|
|
/* FIXME: need to clip client shape to client container
|
|
* with offset for shading, if shading/shaded
|
|
*/
|
|
twin2 = ecore_x_window_override_new(bd->win, 0, 0,
|
|
bd->w - bd->client_inset.l - bd->client_inset.r,
|
|
bd->h - bd->client_inset.t - bd->client_inset.b);
|
|
x = 0;
|
|
y = 0;
|
|
if ((bd->shading) || (bd->shaded))
|
|
{
|
|
if (bd->shade.dir == E_DIRECTION_UP)
|
|
y = bd->h - bd->client_inset.t - bd->client_inset.b - bd->client.h;
|
|
else if (bd->shade.dir == E_DIRECTION_LEFT)
|
|
x = bd->w - bd->client_inset.l - bd->client_inset.r - bd->client.w;
|
|
}
|
|
ecore_x_window_shape_window_set_xy(twin2, bd->client.win,
|
|
x, y);
|
|
ecore_x_window_shape_rectangle_clip(twin2, 0, 0,
|
|
bd->w - bd->client_inset.l - bd->client_inset.r,
|
|
bd->h - bd->client_inset.t - bd->client_inset.b);
|
|
ecore_x_window_shape_window_add_xy(twin, twin2,
|
|
bd->client_inset.l,
|
|
bd->client_inset.t);
|
|
ecore_x_window_del(twin2);
|
|
ecore_x_window_shape_window_set(bd->win, twin);
|
|
ecore_x_window_del(twin);
|
|
}
|
|
else
|
|
{
|
|
ecore_x_window_shape_mask_set(bd->win, 0);
|
|
}
|
|
bd->need_shape_merge = 0;
|
|
}
|
|
|
|
if (bd->need_shape_export)
|
|
{
|
|
Ecore_X_Rectangle *rects;
|
|
int num;
|
|
|
|
rects = ecore_x_window_shape_rectangles_get(bd->win, &num);
|
|
if (rects)
|
|
{
|
|
if (bd->client.shaped)
|
|
e_container_shape_solid_rect_set(bd->shape, 0, 0, 0, 0);
|
|
else
|
|
e_container_shape_solid_rect_set(bd->shape, bd->client_inset.l, bd->client_inset.t, bd->client.w, bd->client.h);
|
|
e_container_shape_rects_set(bd->shape, rects, num);
|
|
free(rects);
|
|
}
|
|
bd->need_shape_export = 0;
|
|
}
|
|
|
|
bd->changed = 0;
|
|
|
|
bd->changes.stack = 0;
|
|
bd->changes.prop = 0;
|
|
bd->changes.border = 0;
|
|
}
|
|
|
|
static void
|
|
_e_border_resize_limit(E_Border *bd, int *w, int *h)
|
|
{
|
|
double a;
|
|
|
|
*w -= bd->client_inset.l + bd->client_inset.r;
|
|
*h -= bd->client_inset.t + bd->client_inset.b;
|
|
if (*h < 1) *h = 1;
|
|
if (*w < 1) *w = 1;
|
|
if ((bd->client.icccm.base_w >= 0) &&
|
|
(bd->client.icccm.base_h >= 0))
|
|
{
|
|
int tw, th;
|
|
|
|
tw = *w - bd->client.icccm.base_w;
|
|
th = *h - bd->client.icccm.base_h;
|
|
if (tw < 1) tw = 1;
|
|
if (th < 1) th = 1;
|
|
a = (double)(tw) / (double)(th);
|
|
if ((bd->client.icccm.min_aspect != 0.0) &&
|
|
(a < bd->client.icccm.min_aspect))
|
|
{
|
|
tw = th * bd->client.icccm.min_aspect;
|
|
*w = tw + bd->client.icccm.base_w;
|
|
}
|
|
else if ((bd->client.icccm.max_aspect != 0.0) &&
|
|
(a > bd->client.icccm.max_aspect))
|
|
{
|
|
th = tw / bd->client.icccm.max_aspect;
|
|
*h = th + bd->client.icccm.base_h;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
a = (double)*w / (double)*h;
|
|
if ((bd->client.icccm.min_aspect != 0.0) &&
|
|
(a < bd->client.icccm.min_aspect))
|
|
*w = *h * bd->client.icccm.min_aspect;
|
|
else if
|
|
((bd->client.icccm.max_aspect != 0.0) &&
|
|
(a > bd->client.icccm.max_aspect))
|
|
*h = *w / bd->client.icccm.max_aspect;
|
|
}
|
|
if (bd->client.icccm.base_w >= 0)
|
|
*w = bd->client.icccm.base_w +
|
|
(((*w - bd->client.icccm.base_w) / bd->client.icccm.step_w) *
|
|
bd->client.icccm.step_w);
|
|
else
|
|
*w = bd->client.icccm.min_w +
|
|
(((*w - bd->client.icccm.min_w) / bd->client.icccm.step_w) *
|
|
bd->client.icccm.step_w);
|
|
if (bd->client.icccm.base_h >= 0)
|
|
*h = bd->client.icccm.base_h +
|
|
(((*h - bd->client.icccm.base_h) / bd->client.icccm.step_h) *
|
|
bd->client.icccm.step_h);
|
|
else
|
|
*h = bd->client.icccm.min_h +
|
|
(((*h - bd->client.icccm.min_h) / bd->client.icccm.step_h) *
|
|
bd->client.icccm.step_h);
|
|
|
|
if (*h < 1) *h = 1;
|
|
if (*w < 1) *w = 1;
|
|
|
|
if (*w > bd->client.icccm.max_w) *w = bd->client.icccm.max_w;
|
|
else if (*w < bd->client.icccm.min_w) *w = bd->client.icccm.min_w;
|
|
if (*h > bd->client.icccm.max_h) *h = bd->client.icccm.max_h;
|
|
else if (*h < bd->client.icccm.min_h) *h = bd->client.icccm.min_h;
|
|
|
|
*w += bd->client_inset.l + bd->client_inset.r;
|
|
*h += bd->client_inset.t + bd->client_inset.b;
|
|
}
|
|
|
|
static void
|
|
_e_border_moveinfo_gather(E_Border *bd, const char *source)
|
|
{
|
|
if (!strcmp(source, "mouse,1")) bd->moveinfo.down.button = 1;
|
|
else if (!strcmp(source, "mouse,2")) bd->moveinfo.down.button = 2;
|
|
else if (!strcmp(source, "mouse,3")) bd->moveinfo.down.button = 3;
|
|
else bd->moveinfo.down.button = 0;
|
|
if ((bd->moveinfo.down.button >= 1) && (bd->moveinfo.down.button <= 3))
|
|
{
|
|
bd->moveinfo.down.mx = bd->mouse.last_down[bd->moveinfo.down.button - 1].mx;
|
|
bd->moveinfo.down.my = bd->mouse.last_down[bd->moveinfo.down.button - 1].my;
|
|
}
|
|
else
|
|
{
|
|
bd->moveinfo.down.mx = bd->mouse.current.mx;
|
|
bd->moveinfo.down.my = bd->mouse.current.my;
|
|
}
|
|
}
|
|
|
|
static void
|
|
_e_border_resize_handle(E_Border *bd)
|
|
{
|
|
int x, y, w, h;
|
|
int new_x, new_y, new_w, new_h;
|
|
int tw, th;
|
|
Evas_List *skiplist = NULL;
|
|
|
|
x = bd->x;
|
|
y = bd->y;
|
|
w = bd->w;
|
|
h = bd->h;
|
|
|
|
if ((bd->resize_mode == RESIZE_TR) ||
|
|
(bd->resize_mode == RESIZE_R) ||
|
|
(bd->resize_mode == RESIZE_BR))
|
|
{
|
|
if ((bd->moveinfo.down.button >= 1) &&
|
|
(bd->moveinfo.down.button <= 3))
|
|
w = bd->mouse.last_down[bd->moveinfo.down.button - 1].w +
|
|
(bd->mouse.current.mx - bd->moveinfo.down.mx);
|
|
else
|
|
w = bd->w + (bd->mouse.current.mx - bd->moveinfo.down.mx);
|
|
}
|
|
else if ((bd->resize_mode == RESIZE_TL) ||
|
|
(bd->resize_mode == RESIZE_L) ||
|
|
(bd->resize_mode == RESIZE_BL))
|
|
{
|
|
if ((bd->moveinfo.down.button >= 1) &&
|
|
(bd->moveinfo.down.button <= 3))
|
|
w = bd->mouse.last_down[bd->moveinfo.down.button - 1].w -
|
|
(bd->mouse.current.mx - bd->moveinfo.down.mx);
|
|
else
|
|
w = bd->w - (bd->mouse.current.mx - bd->moveinfo.down.mx);
|
|
}
|
|
|
|
if ((bd->resize_mode == RESIZE_TL) ||
|
|
(bd->resize_mode == RESIZE_T) ||
|
|
(bd->resize_mode == RESIZE_TR))
|
|
{
|
|
if ((bd->moveinfo.down.button >= 1) &&
|
|
(bd->moveinfo.down.button <= 3))
|
|
h = bd->mouse.last_down[bd->moveinfo.down.button - 1].h -
|
|
(bd->mouse.current.my - bd->moveinfo.down.my);
|
|
else
|
|
h = bd->h - (bd->mouse.current.my - bd->moveinfo.down.my);
|
|
}
|
|
else if ((bd->resize_mode == RESIZE_BL) ||
|
|
(bd->resize_mode == RESIZE_B) ||
|
|
(bd->resize_mode == RESIZE_BR))
|
|
{
|
|
if ((bd->moveinfo.down.button >= 1) &&
|
|
(bd->moveinfo.down.button <= 3))
|
|
h = bd->mouse.last_down[bd->moveinfo.down.button - 1].h +
|
|
(bd->mouse.current.my - bd->moveinfo.down.my);
|
|
else
|
|
h = bd->h + (bd->mouse.current.my - bd->moveinfo.down.my);
|
|
}
|
|
|
|
tw = bd->w;
|
|
th = bd->h;
|
|
|
|
if ((bd->resize_mode == RESIZE_TL) ||
|
|
(bd->resize_mode == RESIZE_L) ||
|
|
(bd->resize_mode == RESIZE_BL))
|
|
x += (tw - w);
|
|
if ((bd->resize_mode == RESIZE_TL) ||
|
|
(bd->resize_mode == RESIZE_T) ||
|
|
(bd->resize_mode == RESIZE_TR))
|
|
y += (th - h);
|
|
|
|
skiplist = evas_list_append(skiplist, bd);
|
|
e_resist_container_border_position(bd->zone->container, skiplist,
|
|
bd->x, bd->y, bd->w, bd->h,
|
|
x, y, w, h,
|
|
&new_x, &new_y, &new_w, &new_h);
|
|
evas_list_free(skiplist);
|
|
|
|
w = new_w;
|
|
h = new_h;
|
|
_e_border_resize_limit(bd, &new_w, &new_h);
|
|
if ((bd->resize_mode == RESIZE_TL) ||
|
|
(bd->resize_mode == RESIZE_L) ||
|
|
(bd->resize_mode == RESIZE_BL))
|
|
new_x += (w - new_w);
|
|
if ((bd->resize_mode == RESIZE_TL) ||
|
|
(bd->resize_mode == RESIZE_T) ||
|
|
(bd->resize_mode == RESIZE_TR))
|
|
new_y += (h - new_h);
|
|
|
|
e_border_move_resize(bd, new_x, new_y, new_w, new_h);
|
|
}
|
|
|
|
static int
|
|
_e_border_shade_animator(void *data)
|
|
{
|
|
E_Border *bd = data;
|
|
double dt, val;
|
|
double dur = bd->client.h / e_config->border_shade_speed;
|
|
|
|
dt = ecore_time_get() - bd->shade.start;
|
|
val = dt / dur;
|
|
|
|
if (val < 0.0) val = 0.0;
|
|
else if (val > 1.0) val = 1.0;
|
|
|
|
if (e_config->border_shade_transition == E_TRANSITION_SINUSOIDAL)
|
|
{
|
|
if (bd->shaded)
|
|
bd->shade.val = (1 - cos(val * M_PI)) / 2.0;
|
|
else
|
|
bd->shade.val = 0.5 + (cos(val * M_PI) / 2.0);
|
|
}
|
|
else if (e_config->border_shade_transition == E_TRANSITION_DECELERATE)
|
|
{
|
|
if (bd->shaded)
|
|
bd->shade.val = sin(val * M_PI / 2.0);
|
|
else
|
|
bd->shade.val = 1 - sin(val * M_PI / 2.0);
|
|
}
|
|
else if (e_config->border_shade_transition == E_TRANSITION_ACCELERATE)
|
|
{
|
|
if (bd->shaded)
|
|
bd->shade.val = 1 - cos(val * M_PI / 2.0);
|
|
else
|
|
bd->shade.val = cos(val * M_PI / 2.0);
|
|
}
|
|
else /* LINEAR if none of the others */
|
|
{
|
|
if (bd->shaded)
|
|
bd->shade.val = val;
|
|
else
|
|
bd->shade.val = 1 - val;
|
|
}
|
|
|
|
/* due to M_PI's innacuracy, cos(M_PI/2) != 0.0, so we need this */
|
|
if (bd->shade.val < 0.001) bd->shade.val = 0.0;
|
|
else if (bd->shade.val > .999) bd->shade.val = 1.0;
|
|
|
|
if (bd->shade.dir == E_DIRECTION_UP)
|
|
{
|
|
bd->h = bd->client_inset.t + bd->client_inset.b + bd->client.h * bd->shade.val;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_DOWN)
|
|
{
|
|
bd->h = bd->client_inset.t + bd->client_inset.b + bd->client.h * bd->shade.val;
|
|
bd->y = bd->shade.y + bd->client.h * (1 - bd->shade.val);
|
|
bd->changes.pos = 1;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_LEFT)
|
|
{
|
|
bd->w = bd->client_inset.l + bd->client_inset.r + bd->client.w * bd->shade.val;
|
|
}
|
|
else if (bd->shade.dir == E_DIRECTION_RIGHT)
|
|
{
|
|
bd->w = bd->client_inset.l + bd->client_inset.r + bd->client.w * bd->shade.val;
|
|
bd->x = bd->shade.x + bd->client.w * (1 - bd->shade.val);
|
|
bd->changes.pos = 1;
|
|
}
|
|
|
|
if ((bd->shaped) || (bd->client.shaped))
|
|
{
|
|
bd->need_shape_merge = 1;
|
|
bd->need_shape_export = 1;
|
|
}
|
|
bd->changes.size = 1;
|
|
bd->changed = 1;
|
|
|
|
/* we're done */
|
|
if ( (bd->shaded && (bd->shade.val == 1)) ||
|
|
(!(bd->shaded) && (bd->shade.val == 0)) )
|
|
{
|
|
E_Event_Border_Resize *ev;
|
|
|
|
bd->shading = 0;
|
|
bd->shaded = !(bd->shaded);
|
|
bd->changes.size = 1;
|
|
bd->changes.shaded = 1;
|
|
bd->changes.shading = 1;
|
|
bd->changed = 1;
|
|
|
|
if (bd->shaded)
|
|
{
|
|
edje_object_signal_emit(bd->bg_object, "shaded", "");
|
|
}
|
|
else
|
|
{
|
|
edje_object_signal_emit(bd->bg_object, "unshaded", "");
|
|
}
|
|
ecore_x_window_gravity_set(bd->client.win, ECORE_X_GRAVITY_NW);
|
|
ev = calloc(1, sizeof(E_Event_Border_Resize));
|
|
ev->border = bd;
|
|
e_object_ref(E_OBJECT(bd));
|
|
ecore_event_add(E_EVENT_BORDER_RESIZE, ev, _e_border_event_border_resize_free, NULL);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
_e_border_cb_border_menu_end(void *data, E_Menu *m)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = e_object_data_get(E_OBJECT(m));
|
|
if (bd) bd->border_menu = NULL;
|
|
e_object_del(E_OBJECT(m));
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_show(E_Border *bd, Evas_Coord x, Evas_Coord y, int key)
|
|
{
|
|
E_Menu *m;
|
|
E_Menu_Item *mi;
|
|
E_App *a;
|
|
|
|
if (bd->border_menu) return;
|
|
|
|
m = e_menu_new();
|
|
e_object_data_set(E_OBJECT(m), bd);
|
|
bd->border_menu = m;
|
|
e_menu_post_deactivate_callback_set(m, _e_border_cb_border_menu_end, NULL);
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Close"));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_close, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/close"),
|
|
"widgets/border/default/close");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Iconify"));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_iconify, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/minimize"),
|
|
"widgets/border/default/minimize");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_separator_set(mi, 1);
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Shaded"));
|
|
e_menu_item_check_set(mi, 1);
|
|
e_menu_item_toggle_set(mi, (bd->shaded ? 1 : 0));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_shade, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/shade"),
|
|
"widgets/border/default/shade");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Maximized"));
|
|
e_menu_item_check_set(mi, 1);
|
|
e_menu_item_toggle_set(mi, (bd->maximized ? 1 : 0));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_maximize, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/maximize"),
|
|
"widgets/border/default/maximize");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Sticky"));
|
|
e_menu_item_check_set(mi, 1);
|
|
e_menu_item_toggle_set(mi, (bd->sticky ? 1 : 0));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_stick, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/stick"),
|
|
"widgets/border/default/stick");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Always On Top"));
|
|
e_menu_item_check_set(mi, 1);
|
|
e_menu_item_toggle_set(mi, (bd->layer == 150 ? 1 : 0));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_on_top, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/on_top"),
|
|
"widgets/border/default/on_top");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_separator_set(mi, 1);
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Send To"));
|
|
e_menu_item_submenu_pre_callback_set(mi, _e_border_menu_sendto_pre_cb, bd);
|
|
e_menu_item_icon_edje_set(mi,
|
|
(char *)e_theme_edje_file_get("base/theme/borders",
|
|
"widgets/border/default/sendto"),
|
|
"widgets/border/default/sendto");
|
|
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_separator_set(mi, 1);
|
|
|
|
a = e_app_window_name_class_find(bd->client.icccm.name,
|
|
bd->client.icccm.class);
|
|
|
|
if (a)
|
|
{
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Edit Icon"));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_icon_edit, a->path);
|
|
e_menu_item_icon_edje_set(mi, a->path, "icon");
|
|
}
|
|
else if (bd->client.icccm.class) /* icons with no class useless to borders */
|
|
{
|
|
static char buf[PATH_MAX + 50];
|
|
char *name, *homedir;
|
|
int i, l;
|
|
|
|
buf[0] = '\0';
|
|
/* generate a reasonable file name from the window class */
|
|
/* FIXME - I think there could be duplicates - how better to do this? */
|
|
name = strdup(bd->client.icccm.class);
|
|
l = strlen(name);
|
|
for (i = 0; i < l; i++)
|
|
{
|
|
if (name[i] == ' ') name[i] = '_';
|
|
}
|
|
/* previously this could be null, but it will exist now */
|
|
homedir = e_user_homedir_get();
|
|
|
|
snprintf(buf, sizeof(buf),
|
|
"--win-class \"%s\" %s/.e/e/applications/all/%s.eapp",
|
|
bd->client.icccm.class, homedir, name);
|
|
mi = e_menu_item_new(m);
|
|
e_menu_item_label_set(mi, _("Create Icon"));
|
|
e_menu_item_callback_set(mi, _e_border_menu_cb_icon_edit, buf);
|
|
}
|
|
|
|
if (key)
|
|
e_menu_activate_key(m, bd->zone, x, y, 1, 1,
|
|
E_MENU_POP_DIRECTION_DOWN);
|
|
else
|
|
e_menu_activate_mouse(m, bd->zone, x, y, 1, 1,
|
|
E_MENU_POP_DIRECTION_DOWN);
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_close(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
if (bd->client.icccm.delete_request)
|
|
ecore_x_window_delete_request_send(bd->client.win);
|
|
else
|
|
{
|
|
ecore_x_kill(bd->client.win);
|
|
ecore_x_sync();
|
|
// ecore_x_window_del(bd->client.win);
|
|
e_border_hide(bd, 0);
|
|
e_object_del(E_OBJECT(bd));
|
|
}
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_iconify(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
if (bd->iconic) e_border_uniconify(bd);
|
|
else e_border_iconify(bd);
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_maximize(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
if (bd->maximized) e_border_unmaximize(bd);
|
|
else e_border_maximize(bd);
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_shade(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
if (bd->shaded) e_border_unshade(bd, E_DIRECTION_UP);
|
|
else e_border_shade(bd, E_DIRECTION_UP);
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_icon_edit(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
char *file;
|
|
char *command;
|
|
char *full;
|
|
Ecore_Exe *process;
|
|
|
|
file = data;
|
|
command = "e_util_eapp_edit ";
|
|
full = malloc(strlen(file) + strlen(command) + 1);
|
|
strcpy(full, command);
|
|
strcat(full, file);
|
|
printf("EXEC %s\n", full);
|
|
process = ecore_exe_run(full, NULL);
|
|
if (!process || !ecore_exe_pid_get(process))
|
|
e_error_dialog_show(_("Icon Edit Error"),
|
|
_("Error starting icon editor\n\n"
|
|
"please install e_util_eapp_edit\n"
|
|
"or make sure it is in your PATH\n"));
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_stick(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
if (bd->sticky) e_border_unstick(bd);
|
|
else e_border_stick(bd);
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_cb_on_top(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
/* FIXME:
|
|
* - Remember old layer
|
|
*/
|
|
E_Border *bd;
|
|
|
|
bd = data;
|
|
if (bd->layer == 150)
|
|
{
|
|
bd->layer = 100;
|
|
e_hints_window_stacking_set(bd, E_STACKING_NONE);
|
|
}
|
|
else
|
|
{
|
|
bd->layer = 150;
|
|
e_hints_window_stacking_set(bd, E_STACKING_ABOVE);
|
|
}
|
|
e_container_border_raise(bd);
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_sendto_pre_cb(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Menu *subm;
|
|
E_Menu_Item *submi;
|
|
E_Border *bd;
|
|
int i;
|
|
|
|
bd = data;
|
|
|
|
subm = e_menu_new();
|
|
e_object_data_set(E_OBJECT(subm), bd);
|
|
e_menu_item_submenu_set(mi, subm);
|
|
|
|
for (i = 0; i < bd->zone->desk_x_count * bd->zone->desk_y_count; i++)
|
|
{
|
|
E_Desk *desk;
|
|
|
|
desk = bd->zone->desks[i];
|
|
submi = e_menu_item_new(subm);
|
|
e_menu_item_label_set(submi, desk->name);
|
|
e_menu_item_callback_set(submi, _e_border_menu_sendto_cb, desk);
|
|
}
|
|
}
|
|
|
|
static void
|
|
_e_border_menu_sendto_cb(void *data, E_Menu *m, E_Menu_Item *mi)
|
|
{
|
|
E_Desk *desk;
|
|
E_Border *bd;
|
|
|
|
desk = data;
|
|
bd = e_object_data_get(E_OBJECT(m));
|
|
if ((bd) && (desk) && (bd->desk != desk))
|
|
{
|
|
e_border_desk_set(bd, desk);
|
|
e_border_hide(bd, 1);
|
|
}
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_resize_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Resize *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_move_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Move *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_add_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Add *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_remove_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Remove *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_show_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Show *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_hide_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Hide *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_iconify_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Iconify *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_uniconify_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Uniconify *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_stick_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Stick *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_unstick_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Unstick *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_zone_set_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Zone_Set *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
e_object_unref(E_OBJECT(e->zone));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_desk_set_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Desk_Set *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
e_object_unref(E_OBJECT(e->desk));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_raise_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Raise *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
if (e->above) e_object_unref(E_OBJECT(e->above));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_lower_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Lower *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
if (e->below) e_object_unref(E_OBJECT(e->below));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_event_border_icon_change_free(void *data, void *ev)
|
|
{
|
|
E_Event_Border_Icon_Change *e;
|
|
|
|
e = ev;
|
|
e_object_unref(E_OBJECT(e->border));
|
|
free(e);
|
|
}
|
|
|
|
static void
|
|
_e_border_zone_update(E_Border *bd)
|
|
{
|
|
E_Container *con;
|
|
Evas_List *l;
|
|
|
|
/* still within old zone - leave it there */
|
|
if (E_INTERSECTS(bd->x, bd->y, bd->w, bd->h,
|
|
bd->zone->x, bd->zone->y, bd->zone->w, bd->zone->h))
|
|
return;
|
|
/* find a new zone */
|
|
con = bd->zone->container;
|
|
for (l = con->zones; l; l = l->next)
|
|
{
|
|
E_Zone *zone;
|
|
|
|
zone = l->data;
|
|
if (E_INTERSECTS(bd->x, bd->y, bd->w, bd->h,
|
|
zone->x, zone->y, zone->w, zone->h))
|
|
{
|
|
e_border_zone_set(bd, zone);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
_e_border_desk_update(E_Border *bd)
|
|
{
|
|
e_border_desk_set(bd, e_desk_current_get(bd->zone));
|
|
}
|
|
|
|
static void
|
|
_e_border_resize_begin(E_Border *bd)
|
|
{
|
|
int w, h;
|
|
|
|
if ((bd->client.icccm.base_w >= 0) &&
|
|
(bd->client.icccm.base_h >= 0))
|
|
{
|
|
w = (bd->client.w - bd->client.icccm.base_w) / bd->client.icccm.step_w;
|
|
h = (bd->client.h - bd->client.icccm.base_h) / bd->client.icccm.step_h;
|
|
}
|
|
else
|
|
{
|
|
w = (bd->client.w - bd->client.icccm.min_w) / bd->client.icccm.step_w;
|
|
h = (bd->client.h - bd->client.icccm.min_h) / bd->client.icccm.step_h;
|
|
}
|
|
e_resize_begin(bd->zone, w, h);
|
|
resize = bd;
|
|
}
|
|
|
|
static void
|
|
_e_border_resize_end(E_Border *bd)
|
|
{
|
|
e_resize_end();
|
|
resize = NULL;
|
|
}
|
|
|
|
static void
|
|
_e_border_resize_update(E_Border *bd)
|
|
{
|
|
int w, h;
|
|
|
|
if ((bd->client.icccm.base_w >= 0) &&
|
|
(bd->client.icccm.base_h >= 0))
|
|
{
|
|
w = (bd->client.w - bd->client.icccm.base_w) / bd->client.icccm.step_w;
|
|
h = (bd->client.h - bd->client.icccm.base_h) / bd->client.icccm.step_h;
|
|
}
|
|
else
|
|
{
|
|
w = (bd->client.w - bd->client.icccm.min_w) / bd->client.icccm.step_w;
|
|
h = (bd->client.h - bd->client.icccm.min_h) / bd->client.icccm.step_h;
|
|
}
|
|
e_resize_update(w, h);
|
|
}
|
|
|
|
static void
|
|
_e_border_move_begin(E_Border *bd)
|
|
{
|
|
e_move_begin(bd->zone, bd->x, bd->y);
|
|
move = bd;
|
|
}
|
|
|
|
static void
|
|
_e_border_move_end(E_Border *bd)
|
|
{
|
|
e_move_end();
|
|
move = NULL;
|
|
}
|
|
|
|
static void
|
|
_e_border_move_update(E_Border *bd)
|
|
{
|
|
e_move_update(bd->x, bd->y);
|
|
}
|
|
|
|
static int
|
|
_e_border_cb_focus_fix(void *data)
|
|
{
|
|
if (!focused)
|
|
{
|
|
/*
|
|
Evas_List *managers;
|
|
E_Manager *man;
|
|
|
|
managers = e_manager_list();
|
|
if (managers)
|
|
{
|
|
E_Container *con;
|
|
|
|
man = managers->data;
|
|
con = e_manager_container_current_get(man);
|
|
if (con)
|
|
{
|
|
printf("set foc to %x [%x]\n",
|
|
man->focus_win, ecore_x_window_focus_get());
|
|
ecore_x_window_focus(man->root);
|
|
}
|
|
}
|
|
*/
|
|
}
|
|
return 1;
|
|
}
|