#include "e.h" #include "e_mod_main.h" #include "e_mod_config.h" #include "e_busycover.h" #define IL_HOME_WIN_TYPE 0xE0b0102f /* local structures */ typedef struct _Il_Home_Win Il_Home_Win; typedef struct _Il_Home_Exec Il_Home_Exec; struct _Il_Home_Win { E_Object e_obj_inherit; E_Win *win; Evas_Object *o_bg, *o_sf, *o_fm, *o_cover; E_Busycover *cover; E_Zone *zone; }; struct _Il_Home_Exec { E_Busycover *cover; Efreet_Desktop *desktop; Ecore_Exe *exec; E_Client *client; E_Zone *zone; Ecore_Timer *timeout; int startup_id; pid_t pid; void *handle; }; /* local function prototypes */ static void _il_home_apps_populate(void); static void _il_home_apps_unpopulate(void); static void _il_home_desks_populate(void); static void _il_home_desktop_run(Il_Home_Win *hwin, Efreet_Desktop *desktop); static E_Client *_il_home_desktop_find_border(E_Zone *zone, Efreet_Desktop *desktop); static void _il_home_win_new(E_Zone *zone); static void _il_home_win_cb_free(Il_Home_Win *hwin); static void _il_home_win_cb_resize(E_Win *win); static void _il_home_fmc_set(Evas_Object *obj); static Eina_Bool _il_home_update_deferred(void *data __UNUSED__); static void _il_home_pan_set(Evas_Object *obj, Evas_Coord x, Evas_Coord y); static void _il_home_pan_get(Evas_Object *obj, Evas_Coord *x, Evas_Coord *y); static void _il_home_pan_max_get(Evas_Object *obj, Evas_Coord *x, Evas_Coord *y); static void _il_home_pan_child_size_get(Evas_Object *obj, Evas_Coord *w, Evas_Coord *h); static void _il_home_cb_selected(void *data, Evas_Object *obj __UNUSED__, void *event __UNUSED__); static Eina_Bool _il_home_desktop_cache_update(void *data __UNUSED__, int type __UNUSED__, void *event __UNUSED__); static Eina_Bool _il_home_cb_border_add(void *data __UNUSED__, int type __UNUSED__, void *event); static Eina_Bool _il_home_cb_border_del(void *data __UNUSED__, int type __UNUSED__, void *event); static Eina_Bool _il_home_cb_exe_del(void *data __UNUSED__, int type __UNUSED__, void *event); static Eina_Bool _il_home_cb_exe_timeout(void *data); static Eina_Bool _il_home_cb_client_message(void *data __UNUSED__, int type __UNUSED__, void *event); static Eina_Bool _il_home_cb_prop_change(void *data __UNUSED__, int type __UNUSED__, void *event); static Eina_Bool _il_home_cb_bg_change(void *data __UNUSED__, int type __UNUSED__, void *event); /* local variables */ static Eina_List *hwins = NULL; static Eina_List *hdls = NULL; static Eina_List *desks = NULL; static Eina_List *exes = NULL; static Ecore_Timer *defer = NULL; /* public functions */ EAPI E_Module_Api e_modapi = { E_MODULE_API_VERSION, "Illume Home" }; EAPI void * e_modapi_init(E_Module *m) { const Eina_List *l; E_Comp *comp; if (!il_home_config_init(m)) return NULL; _il_home_apps_unpopulate(); _il_home_apps_populate(); hdls = eina_list_append(hdls, ecore_event_handler_add(EFREET_EVENT_DESKTOP_CACHE_UPDATE, _il_home_desktop_cache_update, NULL)); hdls = eina_list_append(hdls, ecore_event_handler_add(E_EVENT_CLIENT_ADD, _il_home_cb_border_add, NULL)); hdls = eina_list_append(hdls, ecore_event_handler_add(E_EVENT_CLIENT_REMOVE, _il_home_cb_border_del, NULL)); hdls = eina_list_append(hdls, ecore_event_handler_add(ECORE_EXE_EVENT_DEL, _il_home_cb_exe_del, NULL)); hdls = eina_list_append(hdls, ecore_event_handler_add(ECORE_X_EVENT_CLIENT_MESSAGE, _il_home_cb_client_message, NULL)); hdls = eina_list_append(hdls, ecore_event_handler_add(ECORE_X_EVENT_WINDOW_PROPERTY, _il_home_cb_prop_change, NULL)); hdls = eina_list_append(hdls, ecore_event_handler_add(E_EVENT_BG_UPDATE, _il_home_cb_bg_change, NULL)); EINA_LIST_FOREACH(e_comp_list(), l, comp) { E_Zone *zone; Eina_List *zl; EINA_LIST_FOREACH(comp->zones, zl, zone) { Ecore_X_Illume_Mode mode; mode = ecore_x_e_illume_mode_get(zone->black_win); _il_home_win_new(zone); if (mode > ECORE_X_ILLUME_MODE_SINGLE) _il_home_win_new(zone); } } return m; } EAPI int e_modapi_shutdown(E_Module *m __UNUSED__) { Ecore_Event_Handler *hdl; Il_Home_Win *hwin; Il_Home_Exec *exe; EINA_LIST_FREE(hwins, hwin) e_object_del(E_OBJECT(hwin)); EINA_LIST_FREE(exes, exe) { if (exe->exec) { ecore_exe_terminate(exe->exec); ecore_exe_free(exe->exec); } if (exe->handle) e_busycover_pop(exe->cover, exe->handle); if (exe->timeout) ecore_timer_del(exe->timeout); if (exe->desktop) efreet_desktop_free(exe->desktop); E_FREE(exe); } EINA_LIST_FREE(hdls, hdl) ecore_event_handler_del(hdl); _il_home_apps_unpopulate(); il_home_config_shutdown(); return 1; } EAPI int e_modapi_save(E_Module *m __UNUSED__) { return il_home_config_save(); } void il_home_win_cfg_update(void) { _il_home_apps_unpopulate(); _il_home_apps_populate(); } /* local function prototypes */ static void _il_home_apps_populate(void) { Il_Home_Win *hwin; Eina_List *l; char buff[PATH_MAX]; e_user_dir_concat_static(buff, "appshadow"); ecore_file_mkpath(buff); _il_home_desks_populate(); EINA_LIST_FOREACH(hwins, l, hwin) { _il_home_fmc_set(hwin->o_fm); e_fm2_path_set(hwin->o_fm, NULL, buff); } } static void _il_home_apps_unpopulate(void) { Efreet_Desktop *desktop; Eina_List *files; char buff[PATH_MAX], *file; size_t len; EINA_LIST_FREE(desks, desktop) efreet_desktop_free(desktop); len = e_user_dir_concat_static(buff, "appshadow"); if ((len + 2) >= sizeof(buff)) return; files = ecore_file_ls(buff); buff[len] = '/'; len++; EINA_LIST_FREE(files, file) { if (eina_strlcpy(buff + len, file, sizeof(buff) - len) >= sizeof(buff) - len) continue; ecore_file_unlink(buff); free(file); } } static void _il_home_desks_populate(void) { Efreet_Menu *menu, *entry; Eina_List *ml, *settings, *sys, *kbd; Efreet_Desktop *desktop; int num = 0; if (!(menu = efreet_menu_get())) return; settings = efreet_util_desktop_category_list("Settings"); sys = efreet_util_desktop_category_list("System"); kbd = efreet_util_desktop_category_list("Keyboard"); EINA_LIST_FOREACH(menu->entries, ml, entry) { Eina_List *sl; Efreet_Menu *sm; if (entry->type != EFREET_MENU_ENTRY_MENU) continue; EINA_LIST_FOREACH(entry->entries, sl, sm) { char buff[PATH_MAX]; if (sm->type != EFREET_MENU_ENTRY_DESKTOP) continue; if (!(desktop = sm->desktop)) continue; if ((settings) && (sys) && (eina_list_data_find(settings, desktop)) && (eina_list_data_find(sys, desktop))) continue; if ((kbd) && (eina_list_data_find(kbd, desktop))) continue; efreet_desktop_ref(desktop); desks = eina_list_append(desks, desktop); e_user_dir_snprintf(buff, sizeof(buff), "appshadow/%04x.desktop", num); ecore_file_symlink(desktop->orig_path, buff); num++; } } efreet_menu_free(menu); EINA_LIST_FREE(settings, desktop) efreet_desktop_free(desktop); EINA_LIST_FREE(sys, desktop) efreet_desktop_free(desktop); EINA_LIST_FREE(kbd, desktop) efreet_desktop_free(desktop); } static void _il_home_desktop_run(Il_Home_Win *hwin, Efreet_Desktop *desktop) { E_Exec_Instance *eins; Il_Home_Exec *exec; Eina_List *l; E_Client *ec; char buff[4096]; if ((!hwin) || (!desktop) || (!desktop->exec)) return; EINA_LIST_FOREACH(exes, l, exec) { if (exec->desktop != desktop) continue; if ((exec->client) && (exec->client->zone == hwin->zone)) { e_client_uniconify(exec->client); evas_object_raise(exec->client->frame); evas_object_focus_set(exec->client->frame, 1); return; } } if ((ec = _il_home_desktop_find_border(hwin->zone, desktop))) { e_client_uniconify(ec); evas_object_raise(ec->frame); evas_object_focus_set(ec->frame, 1); return; } exec = E_NEW(Il_Home_Exec, 1); if (!exec) return; exec->cover = hwin->cover; eins = e_exec(hwin->zone, desktop, NULL, NULL, "illume-home"); exec->desktop = desktop; exec->zone = hwin->zone; if (eins) { exec->exec = eins->exe; exec->startup_id = eins->startup_id; if (eins->exe) exec->pid = ecore_exe_pid_get(eins->exe); } exec->timeout = ecore_timer_add(2.0, _il_home_cb_exe_timeout, exec); snprintf(buff, sizeof(buff), "Starting %s", desktop->name); exec->handle = e_busycover_push(hwin->cover, buff, NULL); exes = eina_list_append(exes, exec); } static E_Client * _il_home_desktop_find_border(E_Zone *zone, Efreet_Desktop *desktop) { Eina_List *l; E_Client *ec; char *exe = NULL, *p; if (!desktop) return NULL; if (!desktop->exec) return NULL; p = strchr(desktop->exec, ' '); if (!p) exe = strdup(desktop->exec); else { size_t s = p - desktop->exec + 1; exe = calloc(1, s); if (exe) eina_strlcpy(exe, desktop->exec, s); } if (exe) { p = strrchr(exe, '/'); if (p) strcpy(exe, p + 1); } EINA_LIST_FOREACH(zone->comp->clients, l, ec) { if (e_client_util_ignored_get(ec)) continue; if (ec->zone != zone) continue; if (e_exec_startup_id_pid_find(ec->netwm.pid, ec->netwm.startup_id) == desktop) { free(exe); return ec; } if (exe) { if (ec->icccm.command.argv) { char *pp; pp = strrchr(ec->icccm.command.argv[0], '/'); if (!pp) pp = ec->icccm.command.argv[0]; if (!strcmp(exe, pp)) { free(exe); return ec; } } if ((ec->icccm.name) && (!strcasecmp(ec->icccm.name, exe))) { free(exe); return ec; } } } free(exe); return NULL; } static void _il_home_win_new(E_Zone *zone) { Il_Home_Win *hwin; Evas *evas; E_Desk *desk; char buff[PATH_MAX]; const char *bgfile; hwin = E_OBJECT_ALLOC(Il_Home_Win, IL_HOME_WIN_TYPE, _il_home_win_cb_free); if (!hwin) return; hwin->zone = zone; hwin->win = e_win_new(zone->comp); if (!hwin->win) { e_object_del(E_OBJECT(hwin)); return; } hwin->win->data = hwin; e_win_title_set(hwin->win, _("Illume Home")); e_win_name_class_set(hwin->win, "Illume-Home", "Illume-Home"); e_win_resize_callback_set(hwin->win, _il_home_win_cb_resize); e_win_no_remember_set(hwin->win, EINA_TRUE); snprintf(buff, sizeof(buff), "%s/e-module-illume-home.edj", il_home_cfg->mod_dir); evas = e_win_evas_get(hwin->win); desk = e_desk_current_get(zone); if (desk) bgfile = e_bg_file_get(zone->comp->num, zone->num, desk->x, desk->y); else bgfile = e_bg_file_get(zone->comp->num, zone->num, -1, -1); hwin->o_bg = edje_object_add(evas); edje_object_file_set(hwin->o_bg, bgfile, "e/desktop/background"); eina_stringshare_del(bgfile); evas_object_move(hwin->o_bg, 0, 0); evas_object_show(hwin->o_bg); hwin->o_sf = e_scrollframe_add(evas); e_scrollframe_single_dir_set(hwin->o_sf, EINA_TRUE); e_scrollframe_custom_edje_file_set(hwin->o_sf, buff, "modules/illume-home/launcher/scrollview"); evas_object_move(hwin->o_sf, 0, 0); evas_object_show(hwin->o_sf); hwin->o_fm = e_fm2_add(evas); _il_home_fmc_set(hwin->o_fm); evas_object_show(hwin->o_fm); e_user_dir_concat_static(buff, "appshadow"); e_fm2_path_set(hwin->o_fm, NULL, buff); e_fm2_window_object_set(hwin->o_fm, E_OBJECT(hwin->win)); e_scrollframe_extern_pan_set(hwin->o_sf, hwin->o_fm, _il_home_pan_set, _il_home_pan_get, _il_home_pan_max_get, _il_home_pan_child_size_get); evas_object_propagate_events_set(hwin->o_fm, 0); evas_object_smart_callback_add(hwin->o_fm, "selected", _il_home_cb_selected, hwin); hwin->cover = e_busycover_new(hwin->win); e_win_move_resize(hwin->win, zone->x, zone->y, zone->w, (zone->h / 2)); e_win_show(hwin->win); e_client_zone_set(hwin->win->client, zone); if (hwin->win->evas_win) e_drop_xdnd_register_set(hwin->win->evas_win, EINA_TRUE); hwins = eina_list_append(hwins, hwin); } static void _il_home_win_cb_free(Il_Home_Win *hwin) { if (hwin->win->evas_win) e_drop_xdnd_register_set(hwin->win->evas_win, 0); if (hwin->cover) e_object_del(E_OBJECT(hwin->cover)); if (hwin->o_bg) evas_object_del(hwin->o_bg); if (hwin->o_sf) evas_object_del(hwin->o_sf); if (hwin->o_fm) evas_object_del(hwin->o_fm); if (hwin->win) e_object_del(E_OBJECT(hwin->win)); E_FREE(hwin); } static void _il_home_win_cb_resize(E_Win *win) { Il_Home_Win *hwin; if (!(hwin = win->data)) return; if (hwin->o_bg) evas_object_resize(hwin->o_bg, win->w, win->h); if (hwin->o_sf) evas_object_resize(hwin->o_sf, win->w, win->h); if (hwin->cover) e_busycover_resize(hwin->cover, win->w, win->h); } static void _il_home_fmc_set(Evas_Object *obj) { E_Fm2_Config fmc; if (!obj) return; memset(&fmc, 0, sizeof(E_Fm2_Config)); fmc.view.mode = E_FM2_VIEW_MODE_GRID_ICONS; fmc.view.open_dirs_in_place = 1; fmc.view.selector = 0; fmc.view.single_click = il_home_cfg->single_click; fmc.view.single_click_delay = il_home_cfg->single_click_delay; fmc.view.no_subdir_jump = 1; fmc.icon.extension.show = 0; fmc.icon.icon.w = il_home_cfg->icon_size * e_scale / 2.0; fmc.icon.icon.h = il_home_cfg->icon_size * e_scale / 2.0; fmc.icon.fixed.w = il_home_cfg->icon_size * e_scale / 2.0; fmc.icon.fixed.h = il_home_cfg->icon_size * e_scale / 2.0; fmc.list.sort.no_case = 0; fmc.list.sort.dirs.first = 1; fmc.list.sort.dirs.last = 0; fmc.selection.single = 1; fmc.selection.windows_modifiers = 0; e_fm2_config_set(obj, &fmc); } static Eina_Bool _il_home_update_deferred(void *data __UNUSED__) { _il_home_apps_unpopulate(); _il_home_apps_populate(); defer = NULL; return ECORE_CALLBACK_CANCEL; } static void _il_home_pan_set(Evas_Object *obj, Evas_Coord x, Evas_Coord y) { e_fm2_pan_set(obj, x, y); } static void _il_home_pan_get(Evas_Object *obj, Evas_Coord *x, Evas_Coord *y) { e_fm2_pan_get(obj, x, y); } static void _il_home_pan_max_get(Evas_Object *obj, Evas_Coord *x, Evas_Coord *y) { e_fm2_pan_max_get(obj, x, y); } static void _il_home_pan_child_size_get(Evas_Object *obj, Evas_Coord *w, Evas_Coord *h) { e_fm2_pan_child_size_get(obj, w, h); } static void _il_home_cb_selected(void *data, Evas_Object *obj __UNUSED__, void *event __UNUSED__) { Il_Home_Win *hwin; Eina_List *selected; E_Fm2_Icon_Info *ici; if (!(hwin = data)) return; if (!(selected = e_fm2_selected_list_get(hwin->o_fm))) return; EINA_LIST_FREE(selected, ici) { Efreet_Desktop *desktop; if ((!ici) || (!ici->real_link)) continue; if (!(desktop = efreet_desktop_get(ici->real_link))) continue; _il_home_desktop_run(hwin, desktop); } } static Eina_Bool _il_home_desktop_cache_update(void *data __UNUSED__, int type __UNUSED__, void *event __UNUSED__) { if (defer) ecore_timer_del(defer); defer = ecore_timer_add(0.5, _il_home_update_deferred, NULL); return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _il_home_cb_border_add(void *data __UNUSED__, int type __UNUSED__, void *event) { E_Event_Client *ev; Il_Home_Exec *exe; Eina_List *l; ev = event; EINA_LIST_FOREACH(exes, l, exe) { if (!exe->client) { if ((exe->startup_id == ev->ec->netwm.startup_id) || (exe->pid == ev->ec->netwm.pid)) { exe->client = ev->ec; } } if (!exe->client) continue; if (exe->client->zone != exe->zone) { exe->client->zone = exe->zone; exe->client->x = exe->zone->x; exe->client->y = exe->zone->y; exe->client->changes.pos = 1; EC_CHANGED(exe->client); } if (exe->handle) { e_busycover_pop(exe->cover, exe->handle); exe->handle = NULL; } if (exe->timeout) ecore_timer_del(exe->timeout); exe->timeout = NULL; } return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _il_home_cb_border_del(void *data __UNUSED__, int type __UNUSED__, void *event) { E_Event_Client *ev; Il_Home_Exec *exe; Eina_List *l; ev = event; EINA_LIST_FOREACH(exes, l, exe) { if (exe->client == ev->ec) { exe->exec = NULL; if (exe->handle) e_busycover_pop(exe->cover, exe->handle); exe->handle = NULL; exe->client = NULL; exes = eina_list_remove(exes, exe); E_FREE(exe); break; } } return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _il_home_cb_exe_del(void *data __UNUSED__, int type __UNUSED__, void *event) { Il_Home_Exec *exe; Ecore_Exe_Event_Del *ev; Eina_List *l; ev = event; EINA_LIST_FOREACH(exes, l, exe) { if (exe->pid == ev->pid) { if (exe->handle) { e_busycover_pop(exe->cover, exe->handle); exe->handle = NULL; } exes = eina_list_remove_list(exes, l); if (exe->timeout) ecore_timer_del(exe->timeout); if (exe->desktop) efreet_desktop_free(exe->desktop); E_FREE(exe); break; } } return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _il_home_cb_exe_timeout(void *data) { Il_Home_Exec *exe; if (!(exe = data)) return ECORE_CALLBACK_CANCEL; if (exe->handle) e_busycover_pop(exe->cover, exe->handle); exe->handle = NULL; if (!exe->client) { exes = eina_list_remove(exes, exe); if (exe->desktop) efreet_desktop_free(exe->desktop); E_FREE(exe); return ECORE_CALLBACK_CANCEL; } exe->timeout = NULL; return ECORE_CALLBACK_CANCEL; } static Eina_Bool _il_home_cb_client_message(void *data __UNUSED__, int type __UNUSED__, void *event) { Ecore_X_Event_Client_Message *ev; ev = event; if (ev->message_type == ECORE_X_ATOM_E_ILLUME_HOME_NEW) { E_Zone *zone; zone = e_util_zone_window_find(ev->win); if (zone->black_win != ev->win) return ECORE_CALLBACK_PASS_ON; _il_home_win_new(zone); } else if (ev->message_type == ECORE_X_ATOM_E_ILLUME_HOME_DEL) { E_Client *ec; Eina_List *l; Il_Home_Win *hwin; if (!(ec = e_pixmap_find_client(E_PIXMAP_TYPE_X, ev->win))) return ECORE_CALLBACK_PASS_ON; EINA_LIST_FOREACH(hwins, l, hwin) { if (hwin->win->client == ec) { hwins = eina_list_remove_list(hwins, hwins); e_object_del(E_OBJECT(hwin)); break; } } } return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _il_home_cb_prop_change(void *data __UNUSED__, int type __UNUSED__, void *event) { Ecore_X_Event_Window_Property *ev; Il_Home_Win *hwin; Eina_List *l; ev = event; if (ev->atom != ATM_ENLIGHTENMENT_SCALE) return ECORE_CALLBACK_PASS_ON; EINA_LIST_FOREACH(hwins, l, hwin) if (hwin->o_fm) { _il_home_fmc_set(hwin->o_fm); e_fm2_refresh(hwin->o_fm); } return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _il_home_cb_bg_change(void *data __UNUSED__, int type, void *event __UNUSED__) { Il_Home_Win *hwin; Eina_List *l; if (type != E_EVENT_BG_UPDATE) return ECORE_CALLBACK_PASS_ON; EINA_LIST_FOREACH(hwins, l, hwin) { E_Zone *zone; E_Desk *desk; const char *bgfile; zone = hwin->zone; desk = e_desk_current_get(zone); if (desk) bgfile = e_bg_file_get(zone->comp->num, zone->num, desk->x, desk->y); else bgfile = e_bg_file_get(zone->comp->num, zone->num, -1, -1); edje_object_file_set(hwin->o_bg, bgfile, "e/desktop/background"); eina_stringshare_del(bgfile); } return ECORE_CALLBACK_PASS_ON; }