#include "e.h" #define TEXT_NONE_ACTION_EDGE _("") #define TEXT_PRESS_EDGE_SEQUENCE _("Please select an edge," \ "or click Close to abort." \ "You can either specify a delay of this action using " \ "the slider, or make itrespond to edge clicks:" \ ) #define TEXT_NO_PARAMS _("") static void *_create_data(E_Config_Dialog *cfd); static void _free_data(E_Config_Dialog *cfd, E_Config_Dialog_Data *cfdata); static int _basic_apply_data(E_Config_Dialog *cfd, E_Config_Dialog_Data *cfdata); static Evas_Object *_basic_create_widgets(E_Config_Dialog *cfd, Evas *evas, E_Config_Dialog_Data *cfdata); /********* private functions ***************/ static void _fill_actions_list(E_Config_Dialog_Data *cfdata); /**************** Updates ***********/ static void _update_edge_binding_list(E_Config_Dialog_Data *cfdata); static void _update_action_list(E_Config_Dialog_Data *cfdata); static void _update_action_params(E_Config_Dialog_Data *cfdata); static void _update_buttons(E_Config_Dialog_Data *cfdata); /**************** Callbacks *********/ static void _binding_change_cb(void *data); static void _action_change_cb(void *data); static void _delete_all_edge_binding_cb(void *data, void *data2); static void _delete_edge_binding_cb(void *data, void *data2); // static void _restore_edge_binding_defaults_cb(void *data, void *data2); static void _add_edge_binding_cb(void *data, void *data2); static void _modify_edge_binding_cb(void *data, void *data2); /********* Helper *************************/ static char *_edge_binding_text_get(E_Zone_Edge edge, float delay, int mod, int drag_only); static void _auto_apply_changes(E_Config_Dialog_Data *cfdata); static void _find_edge_binding_action(const char *action, const char *params, int *g, int *a, int *n); /********* Sorting ************************/ static int _edge_binding_sort_cb(const void *d1, const void *d2); /**************** grab window *******/ static void _edge_grab_wnd_show(E_Config_Dialog_Data *cfdata); static void _edge_grab_wnd_cb_apply(void *data, E_Dialog *dia); static void _edge_grab_wnd_cb_close(void *data, E_Dialog *dia); static void _edge_grab_wnd_slider_changed_cb(void *data, Evas_Object *obj); static void _edge_grab_wnd_check_changed_cb(void *data, Evas_Object *obj); static void _edge_grab_wnd_selected_edge_cb(void *data, Evas *e, Evas_Object *obj, void *event_info); static void _edge_grab_wnd_selection_apply(E_Config_Dialog_Data *cfdata); struct _E_Config_Dialog_Data { Evas *evas; struct { Eina_List *edge; } binding; struct { const char *binding, *action; char *params; const char *cur; double delay; int click; int drag_only; int button; int cur_act, add; E_Zone_Edge edge; int modifiers; E_Dialog *dia; } locals; struct { Evas_Object *o_add, *o_mod, *o_del, *o_del_all; Evas_Object *o_binding_list, *o_action_list; Evas_Object *o_params, *o_selector, *o_slider, *o_check, *o_check2, *o_button; } gui; const char *params; int fullscreen_flip; int multiscreen_flip; E_Config_Dialog *cfd; }; E_Config_Dialog * e_int_config_edgebindings(Evas_Object *parent EINA_UNUSED, const char *params) { E_Config_Dialog *cfd; E_Config_Dialog_View *v; if (e_config_dialog_find("E", "keyboard_and_mouse/edge_bindings")) return NULL; v = E_NEW(E_Config_Dialog_View, 1); v->create_cfdata = _create_data; v->free_cfdata = _free_data; v->basic.apply_cfdata = _basic_apply_data; v->basic.create_widgets = _basic_create_widgets; v->override_auto_apply = 1; cfd = e_config_dialog_new(NULL, _("Edge Bindings Settings"), "E", "keyboard_and_mouse/edge_bindings", "enlightenment/edges", 0, v, NULL); if ((params) && (params[0])) { cfd->cfdata->params = eina_stringshare_add(params); _add_edge_binding_cb(cfd->cfdata, NULL); } return cfd; } static void _fill_data(E_Config_Dialog_Data *cfdata) { E_Config_Binding_Edge *bi, *bi2; Eina_List *l; cfdata->locals.params = strdup(""); cfdata->locals.action = eina_stringshare_add(""); cfdata->locals.binding = eina_stringshare_add(""); cfdata->locals.cur = NULL; cfdata->locals.dia = NULL; cfdata->locals.delay = 0.3; cfdata->locals.click = 0; cfdata->binding.edge = NULL; EINA_LIST_FOREACH(e_bindings->edge_bindings, l, bi) { if (!bi) continue; bi2 = E_NEW(E_Config_Binding_Edge, 1); bi2->context = bi->context; bi2->edge = bi->edge; bi2->modifiers = bi->modifiers; bi2->any_mod = bi->any_mod; bi2->delay = bi->delay; bi2->drag_only = bi->drag_only; bi2->action = eina_stringshare_ref(bi->action); bi2->params = eina_stringshare_ref(bi->params); cfdata->binding.edge = eina_list_append(cfdata->binding.edge, bi2); } cfdata->fullscreen_flip = e_config->fullscreen_flip; cfdata->multiscreen_flip = e_config->multiscreen_flip; } static void * _create_data(E_Config_Dialog *cfd) { E_Config_Dialog_Data *cfdata; cfdata = E_NEW(E_Config_Dialog_Data, 1); cfdata->cfd = cfd; _fill_data(cfdata); return cfdata; } static void _free_data(E_Config_Dialog *cfd EINA_UNUSED, E_Config_Dialog_Data *cfdata) { E_Config_Binding_Edge *bi; EINA_LIST_FREE(cfdata->binding.edge, bi) { eina_stringshare_del(bi->action); eina_stringshare_del(bi->params); E_FREE(bi); } eina_stringshare_del(cfdata->locals.cur); eina_stringshare_del(cfdata->params); eina_stringshare_del(cfdata->locals.binding); eina_stringshare_del(cfdata->locals.action); free(cfdata->locals.params); E_FREE(cfdata); } static int _basic_apply_data(E_Config_Dialog *cfd EINA_UNUSED, E_Config_Dialog_Data *cfdata) { const Eina_List *l; E_Zone *zone; E_Config_Binding_Edge *bi, *bi2; E_Layer layer; _auto_apply_changes(cfdata); EINA_LIST_FREE(e_bindings->edge_bindings, bi) { e_bindings_edge_del(bi->context, bi->edge, bi->drag_only, bi->modifiers, bi->any_mod, bi->action, bi->params, bi->delay); eina_stringshare_del(bi->action); eina_stringshare_del(bi->params); E_FREE(bi); } e_config->multiscreen_flip = cfdata->multiscreen_flip; EINA_LIST_FOREACH(cfdata->binding.edge, l, bi2) { bi = E_NEW(E_Config_Binding_Edge, 1); bi->context = bi2->context; bi->edge = bi2->edge; bi->modifiers = bi2->modifiers; bi->any_mod = bi2->any_mod; bi->delay = bi2->delay; bi->drag_only = bi2->drag_only; bi->action = ((!bi2->action) || (!bi2->action[0])) ? NULL : eina_stringshare_add(bi2->action); bi->params = ((!bi2->params) || (!bi2->params[0])) ? NULL : eina_stringshare_add(bi2->params); e_bindings->edge_bindings = eina_list_append(e_bindings->edge_bindings, bi); e_bindings_edge_add(bi->context, bi->edge, bi->drag_only, bi->modifiers, bi->any_mod, bi->action, bi->params, bi->delay); } if (cfdata->fullscreen_flip != e_config->fullscreen_flip) { if (cfdata->fullscreen_flip) layer = E_LAYER_CLIENT_EDGE_FULLSCREEN; else layer = E_LAYER_CLIENT_EDGE; EINA_LIST_FOREACH(e_comp->zones, l, zone) e_zone_edge_win_layer_set(zone, layer); } e_config->fullscreen_flip = cfdata->fullscreen_flip; e_config_save_queue(); return 1; } static Evas_Object * _basic_create_widgets(E_Config_Dialog *cfd, Evas *evas, E_Config_Dialog_Data *cfdata) { Evas_Object *o, *ol, *ot, *of, *ob; cfdata->evas = evas; o = e_widget_list_add(evas, 0, 0); ol = e_widget_list_add(evas, 0, 1); of = e_widget_frametable_add(evas, _("Edge Bindings"), 0); ob = e_widget_ilist_add(evas, 32, 32, &(cfdata->locals.binding)); cfdata->gui.o_binding_list = ob; e_widget_size_min_set(ob, 200, 160); e_widget_frametable_object_append(of, ob, 0, 0, 2, 1, 1, 1, 1, 1); ob = e_widget_button_add(evas, _("Add"), "list-add", _add_edge_binding_cb, cfdata, NULL); cfdata->gui.o_add = ob; e_widget_frametable_object_append(of, ob, 0, 1, 1, 1, 1, 0, 1, 0); ob = e_widget_button_add(evas, _("Delete"), "list-remove", _delete_edge_binding_cb, cfdata, NULL); cfdata->gui.o_del = ob; e_widget_disabled_set(ob, 1); e_widget_frametable_object_append(of, ob, 1, 1, 1, 1, 1, 0, 1, 0); ob = e_widget_button_add(evas, _("Modify"), NULL, _modify_edge_binding_cb, cfdata, NULL); cfdata->gui.o_mod = ob; e_widget_disabled_set(ob, 1); e_widget_frametable_object_append(of, ob, 0, 2, 1, 1, 1, 0, 1, 0); ob = e_widget_button_add(evas, _("Delete All"), "edit-clear", _delete_all_edge_binding_cb, cfdata, NULL); cfdata->gui.o_del_all = ob; e_widget_disabled_set(ob, 1); e_widget_frametable_object_append(of, ob, 1, 2, 1, 1, 1, 0, 1, 0); // ob = e_widget_button_add(evas, _("Restore Default Bindings"), "enlightenment", _restore_edge_binding_defaults_cb, cfdata, NULL); // e_widget_frametable_object_append(of, ob, 0, 3, 2, 1, 1, 0, 1, 0); e_widget_list_object_append(ol, of, 1, 1, 0.5); ot = e_widget_table_add(e_win_evas_win_get(evas), 0); of = e_widget_framelist_add(evas, _("Action"), 0); ob = e_widget_ilist_add(evas, 24, 24, &(cfdata->locals.action)); cfdata->gui.o_action_list = ob; e_widget_size_min_set(ob, 200, 240); e_widget_framelist_object_append(of, ob); e_widget_table_object_append(ot, of, 0, 0, 1, 1, 1, 1, 1, 1); of = e_widget_framelist_add(evas, _("Mouse Button"), 0); ob = e_widget_label_add(evas, NULL); e_widget_framelist_object_append(of, ob); e_widget_table_object_append(ot, of, 0, 2, 1, 1, 1, 1, 1, 0); cfdata->gui.o_button = ob; of = e_widget_framelist_add(evas, _("Action Params"), 0); ob = e_widget_entry_add(cfd->dia->win, &(cfdata->locals.params), NULL, NULL, NULL); cfdata->gui.o_params = ob; e_widget_disabled_set(ob, 1); e_widget_framelist_object_append(of, ob); e_widget_table_object_append(ot, of, 0, 3, 1, 1, 1, 1, 1, 0); e_widget_list_object_append(ol, ot, 1, 1, 0.5); e_widget_list_object_append(o, ol, 1, 1, 0.5); of = e_widget_framelist_add(evas, _("General Options"), 0); ob = e_widget_check_add(evas, _("Allow binding activation with fullscreen windows"), &(cfdata->fullscreen_flip)); e_widget_framelist_object_append(of, ob); ob = e_widget_check_add(evas, _("Allow binding activation with multiple screens (DANGEROUS!)"), &(cfdata->multiscreen_flip)); e_widget_framelist_object_append(of, ob); e_widget_list_object_append(o, of, 1, 0, 0.5); _update_edge_binding_list(cfdata); _fill_actions_list(cfdata); e_dialog_resizable_set(cfd->dia, 1); return o; } static void _fill_actions_list(E_Config_Dialog_Data *cfdata) { char buf[1024]; Eina_List *l, *l2; E_Action_Group *actg; E_Action_Description *actd; int g, a; evas_event_freeze(evas_object_evas_get(cfdata->gui.o_action_list)); edje_freeze(); e_widget_ilist_freeze(cfdata->gui.o_action_list); e_widget_ilist_clear(cfdata->gui.o_action_list); for (l = e_action_groups_get(), g = 0; l; l = l->next, g++) { actg = l->data; if (!actg->acts) continue; e_widget_ilist_header_append(cfdata->gui.o_action_list, NULL, _(actg->act_grp)); for (l2 = actg->acts, a = 0; l2; l2 = l2->next, a++) { actd = l2->data; snprintf(buf, sizeof(buf), "%d %d", g, a); e_widget_ilist_append(cfdata->gui.o_action_list, NULL, _(actd->act_name), _action_change_cb, cfdata, buf); } } e_widget_ilist_go(cfdata->gui.o_action_list); e_widget_ilist_thaw(cfdata->gui.o_action_list); edje_thaw(); evas_event_thaw(evas_object_evas_get(cfdata->gui.o_action_list)); } /**************** Callbacks *********/ static void _add_edge_binding_cb(void *data, void *data2 EINA_UNUSED) { E_Config_Dialog_Data *cfdata; cfdata = data; _auto_apply_changes(cfdata); cfdata->locals.add = 1; _edge_grab_wnd_show(cfdata); } static void _modify_edge_binding_cb(void *data, void *data2 EINA_UNUSED) { E_Config_Dialog_Data *cfdata; cfdata = data; _auto_apply_changes(cfdata); cfdata->locals.add = 0; if (cfdata->locals.cur && cfdata->locals.cur[0]) { E_Config_Binding_Edge *bi; int n; if (sscanf(cfdata->locals.cur, "e%d", &n) != 1) return; bi = eina_list_nth(cfdata->binding.edge, n); cfdata->locals.edge = bi->edge; cfdata->locals.drag_only = bi->drag_only; cfdata->locals.delay = ((double)bi->delay); if (bi->delay <= -1.0) cfdata->locals.click = 1, cfdata->locals.button = -bi->delay; else cfdata->locals.click = 0, cfdata->locals.button = 0; cfdata->locals.modifiers = bi->modifiers; } else return; _edge_grab_wnd_show(cfdata); } static void _binding_change_cb(void *data) { E_Config_Dialog_Data *cfdata; cfdata = data; _auto_apply_changes(cfdata); if (cfdata->locals.cur) eina_stringshare_del(cfdata->locals.cur); cfdata->locals.cur = NULL; if ((!cfdata->locals.binding) || (!cfdata->locals.binding[0])) return; cfdata->locals.cur = eina_stringshare_ref(cfdata->locals.binding); _update_buttons(cfdata); _update_action_list(cfdata); } static void _action_change_cb(void *data) { E_Config_Dialog_Data *cfdata; cfdata = data; _update_action_params(cfdata); } static void _delete_all_edge_binding_cb(void *data, void *data2 EINA_UNUSED) { E_Config_Binding_Edge *bi; E_Config_Dialog_Data *cfdata; cfdata = data; EINA_LIST_FREE(cfdata->binding.edge, bi) { eina_stringshare_del(bi->action); eina_stringshare_del(bi->params); E_FREE(bi); } eina_stringshare_del(cfdata->locals.cur); cfdata->locals.cur = NULL; e_widget_ilist_clear(cfdata->gui.o_binding_list); e_widget_ilist_go(cfdata->gui.o_binding_list); e_widget_ilist_unselect(cfdata->gui.o_action_list); e_widget_entry_clear(cfdata->gui.o_params); e_widget_disabled_set(cfdata->gui.o_params, 1); e_widget_label_text_set(cfdata->gui.o_button, NULL); _update_buttons(cfdata); } static void _delete_edge_binding_cb(void *data, void *data2 EINA_UNUSED) { Eina_List *l = NULL; int sel, n; E_Config_Dialog_Data *cfdata; E_Config_Binding_Edge *bi; cfdata = data; sel = e_widget_ilist_selected_get(cfdata->gui.o_binding_list); if (cfdata->locals.binding[0] == 'e') { if (sscanf(cfdata->locals.binding, "e%d", &n) != 1) return; l = eina_list_nth_list(cfdata->binding.edge, n); if (l) { bi = eina_list_data_get(l); eina_stringshare_del(bi->action); eina_stringshare_del(bi->params); E_FREE(bi); cfdata->binding.edge = eina_list_remove_list(cfdata->binding.edge, l); } } _update_edge_binding_list(cfdata); if (sel >= e_widget_ilist_count(cfdata->gui.o_binding_list)) sel = e_widget_ilist_count(cfdata->gui.o_binding_list) - 1; eina_stringshare_del(cfdata->locals.cur); cfdata->locals.cur = NULL; e_widget_ilist_selected_set(cfdata->gui.o_binding_list, sel); if (sel < 0) { e_widget_ilist_unselect(cfdata->gui.o_action_list); e_widget_entry_clear(cfdata->gui.o_params); e_widget_disabled_set(cfdata->gui.o_params, 1); e_widget_label_text_set(cfdata->gui.o_button, NULL); _update_buttons(cfdata); } }/* static void _restore_edge_binding_defaults_cb(void *data, void *data2 EINA_UNUSED) { E_Config_Bindings *ecb; Eina_Stringshare *prof; E_Config_Dialog_Data *cfdata = data; ecb = e_config_domain_system_load("e_bindings", e_config_descriptor_find("E_Config_Bindings")); if (!ecb) { const char *type; prof = eina_stringshare_ref(e_config_profile_get()); switch (e_config->config_type) { case E_CONFIG_PROFILE_TYPE_DESKTOP: type = "standard"; break; case E_CONFIG_PROFILE_TYPE_MOBILE: type = "mobile"; break; //case E_CONFIG_PROFILE_TYPE_TABLET: FIXME - not used default: type = "default"; break; } e_config_profile_set(type); ecb = e_config_domain_system_load("e_bindings", e_config_descriptor_find("E_Config_Bindings")); e_config_profile_set(prof); eina_stringshare_del(prof); } if (!ecb) return; E_FREE_LIST(cfdata->binding.edge, e_config_binding_edge_free); cfdata->binding.edge = ecb->edge_bindings, ecb->edge_bindings = NULL; e_config_bindings_free(ecb); eina_stringshare_del(cfdata->locals.cur); cfdata->locals.cur = NULL; _update_edge_binding_list(cfdata); _update_buttons(cfdata); e_widget_ilist_unselect(cfdata->gui.o_action_list); e_widget_entry_clear(cfdata->gui.o_params); e_widget_disabled_set(cfdata->gui.o_params, 1); }*/ /**************** Updates ***********/ static void _update_action_list(E_Config_Dialog_Data *cfdata) { E_Config_Binding_Edge *bi; int j = -1, i, n; const char *action, *params; char buf[256]; if (!cfdata->locals.cur) return; if (cfdata->locals.cur[0] == 'e') { if (sscanf(cfdata->locals.cur, "e%d", &n) != 1) return; bi = eina_list_nth(cfdata->binding.edge, n); if (!bi) { e_widget_ilist_unselect(cfdata->gui.o_action_list); e_widget_entry_clear(cfdata->gui.o_params); e_widget_disabled_set(cfdata->gui.o_params, 1); return; } action = bi->action; params = bi->params; } else return; _find_edge_binding_action(action, params, NULL, NULL, &j); if (j >= 0) { for (i = 0; i < e_widget_ilist_count(cfdata->gui.o_action_list); i++) { if (i > j) break; if (e_widget_ilist_nth_is_header(cfdata->gui.o_action_list, i)) j++; } } if (j >= 0) { if (j == e_widget_ilist_selected_get(cfdata->gui.o_action_list)) _update_action_params(cfdata); else e_widget_ilist_selected_set(cfdata->gui.o_action_list, j); } else { e_widget_ilist_unselect(cfdata->gui.o_action_list); eina_stringshare_del(cfdata->locals.action); cfdata->locals.action = eina_stringshare_add(""); e_widget_entry_clear(cfdata->gui.o_params); e_widget_label_text_set(cfdata->gui.o_button, NULL); } if (bi->delay < 0) { snprintf(buf, sizeof(buf), "%d", (int)-bi->delay); e_widget_label_text_set(cfdata->gui.o_button, buf); } else e_widget_label_text_set(cfdata->gui.o_button, NULL); /*if (cfdata->locals.cur[0] == 'e') { sscanf(cfdata->locals.cur, "e%d", &n); bi = eina_list_nth(cfdata->binding.edge, n); if (!bi) { e_widget_ilist_unselect(cfdata->gui.o_action_list); e_widget_entry_clear(cfdata->gui.o_params); e_widget_disabled_set(cfdata->gui.o_params, 1); return; } _find_edge_binding_action(bi, NULL, NULL, &j); if (j >= 0) { for (i = 0; i < e_widget_ilist_count(cfdata->gui.o_action_list); i++) { if (i > j) break; if (e_widget_ilist_nth_is_header(cfdata->gui.o_action_list, i)) j++; } } if (j >= 0) { if (j == e_widget_ilist_selected_get(cfdata->gui.o_action_list)) _update_action_params(cfdata); else e_widget_ilist_selected_set(cfdata->gui.o_action_list, j); } else { e_widget_ilist_unselect(cfdata->gui.o_action_list); free(cfdata->locals.action); cfdata->locals.action = strdup(""); e_widget_entry_clear(cfdata->gui.o_params); } }*/ } static void _update_action_params(E_Config_Dialog_Data *cfdata) { int g, a, b; E_Action_Group *actg; E_Action_Description *actd; E_Config_Binding_Edge *bi; const char *action, *params; #define EDGE_EXAMPLE_PARAMS \ if ((!actd->param_example) || (!actd->param_example[0])) \ e_widget_entry_text_set(cfdata->gui.o_params, TEXT_NO_PARAMS); \ else \ e_widget_entry_text_set(cfdata->gui.o_params, actd->param_example) if ((!cfdata->locals.action) || (!cfdata->locals.action[0])) { e_widget_disabled_set(cfdata->gui.o_params, 1); e_widget_entry_clear(cfdata->gui.o_params); return; } if (sscanf(cfdata->locals.action, "%d %d", &g, &a) != 2) return; actg = eina_list_nth(e_action_groups_get(), g); if (!actg) return; actd = eina_list_nth(actg->acts, a); if (!actd) return; if (actd->act_params) { e_widget_disabled_set(cfdata->gui.o_params, 1); e_widget_entry_text_set(cfdata->gui.o_params, actd->act_params); return; } if ((!cfdata->locals.cur) || (!cfdata->locals.cur[0])) { e_widget_disabled_set(cfdata->gui.o_params, 1); EDGE_EXAMPLE_PARAMS; return; } if (!actd->editable) e_widget_disabled_set(cfdata->gui.o_params, 1); else e_widget_disabled_set(cfdata->gui.o_params, 0); if (cfdata->locals.cur[0] == 'e') { if (sscanf(cfdata->locals.cur, "e%d", &b) != 1) { e_widget_disabled_set(cfdata->gui.o_params, 1); EDGE_EXAMPLE_PARAMS; return; } bi = eina_list_nth(cfdata->binding.edge, b); if (!bi) { e_widget_disabled_set(cfdata->gui.o_params, 1); EDGE_EXAMPLE_PARAMS; return; } action = bi->action; params = bi->params; } else { e_widget_disabled_set(cfdata->gui.o_params, 1); EDGE_EXAMPLE_PARAMS; return; } if (action) { if (!strcmp(action, actd->act_cmd)) { if ((!params) || (!params[0])) EDGE_EXAMPLE_PARAMS; else e_widget_entry_text_set(cfdata->gui.o_params, params); } else EDGE_EXAMPLE_PARAMS; } else EDGE_EXAMPLE_PARAMS; } static void _update_edge_binding_list(E_Config_Dialog_Data *cfdata) { int i; char *b, b2[64]; Eina_List *l; E_Config_Binding_Edge *bi; evas_event_freeze(evas_object_evas_get(cfdata->gui.o_binding_list)); edje_freeze(); e_widget_ilist_freeze(cfdata->gui.o_binding_list); e_widget_ilist_clear(cfdata->gui.o_binding_list); e_widget_ilist_go(cfdata->gui.o_binding_list); if (cfdata->binding.edge) { cfdata->binding.edge = eina_list_sort(cfdata->binding.edge, eina_list_count(cfdata->binding.edge), _edge_binding_sort_cb); } for (l = cfdata->binding.edge, i = 0; l; l = l->next, i++) { Evas_Object *ic; bi = l->data; b = _edge_binding_text_get(bi->edge, bi->delay, bi->modifiers, bi->drag_only); if (!b) continue; ic = edje_object_add(cfdata->evas); e_util_edje_icon_set(ic, "enlightenment/edges"); snprintf(b2, sizeof(b2), "e%d", i); e_widget_ilist_append(cfdata->gui.o_binding_list, ic, b, _binding_change_cb, cfdata, b2); free(b); } e_widget_ilist_go(cfdata->gui.o_binding_list); e_widget_ilist_thaw(cfdata->gui.o_binding_list); edje_thaw(); evas_event_thaw(evas_object_evas_get(cfdata->gui.o_binding_list)); if (eina_list_count(cfdata->binding.edge)) e_widget_disabled_set(cfdata->gui.o_del_all, 0); else e_widget_disabled_set(cfdata->gui.o_del_all, 1); } static void _update_buttons(E_Config_Dialog_Data *cfdata) { if (e_widget_ilist_count(cfdata->gui.o_binding_list)) e_widget_disabled_set(cfdata->gui.o_del_all, 0); else e_widget_disabled_set(cfdata->gui.o_del_all, 1); if (!cfdata->locals.cur) { e_widget_disabled_set(cfdata->gui.o_mod, 1); e_widget_disabled_set(cfdata->gui.o_del, 1); return; } e_widget_disabled_set(cfdata->gui.o_mod, 0); e_widget_disabled_set(cfdata->gui.o_del, 0); } /*************** Sorting *****************************/ static int _edge_binding_sort_cb(const void *d1, const void *d2) { int i, j; const E_Config_Binding_Edge *bi, *bi2; bi = d1; bi2 = d2; i = 0; j = 0; if (bi->modifiers & E_BINDING_MODIFIER_CTRL) i++; if (bi->modifiers & E_BINDING_MODIFIER_ALT) i++; if (bi->modifiers & E_BINDING_MODIFIER_SHIFT) i++; if (bi->modifiers & E_BINDING_MODIFIER_WIN) i++; if (bi2->modifiers & E_BINDING_MODIFIER_CTRL) j++; if (bi2->modifiers & E_BINDING_MODIFIER_ALT) j++; if (bi2->modifiers & E_BINDING_MODIFIER_SHIFT) j++; if (bi2->modifiers & E_BINDING_MODIFIER_WIN) j++; if (i < j) return -1; else if (i > j) return 1; if (bi->modifiers < bi2->modifiers) return -1; else if (bi->modifiers > bi2->modifiers) return 1; if (bi->edge < bi2->edge) return -1; else if (bi->edge > bi2->edge) return 1; return 0; } /**************** grab window *******/ static void _dia_del(void *data) { E_Config_Dialog_Data *cfdata; cfdata = e_object_data_get(data); if (!cfdata) return; cfdata->locals.dia = NULL; } static void _edge_grab_wnd_show(E_Config_Dialog_Data *cfdata) { Evas_Object *o, *obg, *os; Evas *evas; Evas_Coord minw, minh; const char *bgfile; int tw, th; char *label = NULL; if (cfdata->locals.dia != 0) return; cfdata->locals.dia = e_dialog_normal_win_new(cfdata->cfd->dia->win, "E", "_edgebind_getedge_dialog"); if (!cfdata->locals.dia) return; e_dialog_title_set(cfdata->locals.dia, _("Edge Binding Sequence")); e_dialog_button_add(cfdata->locals.dia, _("Apply"), NULL, _edge_grab_wnd_cb_apply, cfdata); e_dialog_button_add(cfdata->locals.dia, _("Close"), NULL, _edge_grab_wnd_cb_close, cfdata); e_object_data_set(E_OBJECT(cfdata->locals.dia), cfdata); e_object_del_attach_func_set(E_OBJECT(cfdata->locals.dia), _dia_del); elm_win_center(cfdata->locals.dia->win, 1, 1); evas = evas_object_evas_get(cfdata->locals.dia->win); cfdata->gui.o_selector = o = edje_object_add(evas); e_theme_edje_object_set(o, "base/theme/modules/conf_edgebindings", "e/modules/conf_edgebindings/selection"); cfdata->gui.o_slider = os = e_widget_slider_add(evas, 1, 0, _("%.2f seconds"), 0.0, 2.0, 0.05, 0, &(cfdata->locals.delay), NULL, 200); edje_object_part_swallow(o, "e.swallow.slider", os); e_widget_on_change_hook_set(os, _edge_grab_wnd_slider_changed_cb, cfdata); evas_object_show(os); cfdata->gui.o_check = os = e_widget_check_add(evas, _("Clickable edge"), &(cfdata->locals.click)); e_widget_size_min_resize(os); e_widget_on_change_hook_set(os, _edge_grab_wnd_check_changed_cb, cfdata); edje_object_part_box_append(o, "e.box", os); evas_object_show(os); e_widget_size_min_get(os, &minw, &minh); evas_object_size_hint_min_set(os, minw, minh); cfdata->gui.o_check2 = os = e_widget_check_add(evas, _("Drag only"), &(cfdata->locals.drag_only)); e_widget_size_min_resize(os); e_widget_on_change_hook_set(os, _edge_grab_wnd_check_changed_cb, cfdata); edje_object_part_box_append(o, "e.box", os); evas_object_show(os); e_widget_size_min_get(os, &minw, &minh); evas_object_size_hint_min_set(os, minw, minh); e_widget_disabled_set(cfdata->gui.o_slider, cfdata->locals.click); e_widget_disabled_set(cfdata->gui.o_check2, cfdata->locals.click); e_widget_disabled_set(cfdata->gui.o_check, cfdata->locals.drag_only); edje_object_part_text_set(o, "e.text.description", TEXT_PRESS_EDGE_SEQUENCE); edje_object_size_min_get(o, &minw, &minh); if (!minw || !minh) { edje_object_calc_force(o); edje_object_size_min_calc(o, &minw, &minh); } e_dialog_content_set(cfdata->locals.dia, o, minw, minh); bgfile = e_bg_file_get(0, 0, 0); obg = e_thumb_icon_add(evas); e_icon_fill_inside_set(obg, 0); e_thumb_icon_file_set(obg, bgfile, "e/desktop/background"); eina_stringshare_del(bgfile); edje_object_part_geometry_get(o, "e.swallow.background", NULL, NULL, &tw, &th); e_thumb_icon_size_set(obg, tw, th); evas_object_size_hint_max_set(obg, tw, th); edje_object_part_swallow(o, "e.swallow.background", obg); e_thumb_icon_begin(obg); evas_object_show(obg); if (cfdata->locals.edge) { label = _edge_binding_text_get(cfdata->locals.edge, ((float)cfdata->locals.delay), cfdata->locals.modifiers, cfdata->locals.drag_only); edje_object_part_text_set(cfdata->gui.o_selector, "e.text.selection", label); E_FREE(label); } else edje_object_part_text_set(cfdata->gui.o_selector, "e.text.selection", _("No edge selected")); evas_object_event_callback_add(o, EVAS_CALLBACK_MOUSE_DOWN, _edge_grab_wnd_selected_edge_cb, cfdata); e_dialog_show(cfdata->locals.dia); } static void _edge_grab_wnd_hide(E_Config_Dialog_Data *cfdata) { e_object_del(E_OBJECT(cfdata->locals.dia)); cfdata->locals.dia = NULL; } static void _edge_grab_wnd_cb_apply(void *data, E_Dialog *dia EINA_UNUSED) { E_Config_Dialog_Data *cfdata; cfdata = data; if (!cfdata->locals.edge) return; _edge_grab_wnd_selection_apply(cfdata); _edge_grab_wnd_hide(cfdata); } static void _edge_grab_wnd_cb_close(void *data, E_Dialog *dia EINA_UNUSED) { E_Config_Dialog_Data *cfdata; cfdata = data; _edge_grab_wnd_hide(cfdata); } static void _edge_grab_wnd_slider_changed_cb(void *data, Evas_Object *obj EINA_UNUSED) { E_Config_Dialog_Data *cfdata = data; char *label = NULL; if (!cfdata->locals.edge) return; label = _edge_binding_text_get(cfdata->locals.edge, ((float)cfdata->locals.delay), cfdata->locals.modifiers, cfdata->locals.drag_only); edje_object_part_text_set(cfdata->gui.o_selector, "e.text.selection", label); E_FREE(label); } static void _edge_grab_wnd_check_changed_cb(void *data, Evas_Object *obj EINA_UNUSED) { E_Config_Dialog_Data *cfdata = data; char *label = NULL; if (cfdata->locals.click) { if (cfdata->locals.edge && cfdata->locals.button) label = _edge_binding_text_get(cfdata->locals.edge, -1.0 * cfdata->locals.button, cfdata->locals.modifiers, cfdata->locals.drag_only); } else { if (cfdata->locals.edge) label = _edge_binding_text_get(cfdata->locals.edge, ((float)cfdata->locals.delay), cfdata->locals.modifiers, cfdata->locals.drag_only); e_widget_disabled_set(cfdata->gui.o_slider, 0); } e_widget_disabled_set(cfdata->gui.o_slider, cfdata->locals.click); e_widget_disabled_set(cfdata->gui.o_check2, cfdata->locals.click); e_widget_disabled_set(cfdata->gui.o_check, cfdata->locals.drag_only); edje_object_part_text_set(cfdata->gui.o_selector, "e.text.selection", label); E_FREE(label); } static void _edge_grab_wnd_selected_edge_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info) { Evas_Event_Mouse_Down *event; E_Config_Dialog_Data *cfdata; E_Zone_Edge edge; Evas_Coord xx, yy, x, y, w, h; char *label; if (!(cfdata = data)) return; if (!(event = event_info)) return; cfdata->locals.button = event->button; evas_object_geometry_get(cfdata->gui.o_selector, &xx, &yy, NULL, NULL); edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.top_left", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_TOP_LEFT; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.top", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_TOP; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.top_right", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_TOP_RIGHT; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.right", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_RIGHT; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.bottom_right", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_BOTTOM_RIGHT; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.bottom", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_BOTTOM; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.bottom_left", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_BOTTOM_LEFT; goto stop; } edje_object_part_geometry_get(cfdata->gui.o_selector, "e.edge.left", &x, &y, &w, &h); if (E_INSIDE(event->canvas.x, event->canvas.y, x + xx, y + yy, w, h)) { edge = E_ZONE_EDGE_LEFT; goto stop; } return; stop: cfdata->locals.edge = edge; cfdata->locals.modifiers = 0; if (evas_key_modifier_is_set(event->modifiers, "Control")) cfdata->locals.modifiers |= E_BINDING_MODIFIER_CTRL; if (evas_key_modifier_is_set(event->modifiers, "Shift")) cfdata->locals.modifiers |= E_BINDING_MODIFIER_SHIFT; if (evas_key_modifier_is_set(event->modifiers, "Alt")) cfdata->locals.modifiers |= E_BINDING_MODIFIER_ALT; if (evas_key_modifier_is_set(event->modifiers, "Win")) cfdata->locals.modifiers |= E_BINDING_MODIFIER_WIN; label = _edge_binding_text_get(cfdata->locals.edge, cfdata->locals.click ? (-1.0 * cfdata->locals.button) : ((float)cfdata->locals.delay), cfdata->locals.modifiers, cfdata->locals.drag_only); edje_object_part_text_set(cfdata->gui.o_selector, "e.text.selection", label); E_FREE(label); } static void _edge_grab_wnd_selection_apply(E_Config_Dialog_Data *cfdata) { E_Config_Binding_Edge *bi = NULL, *bi2 = NULL; Eina_List *l; int found = 0, n = -1; if (cfdata->locals.click) cfdata->locals.delay = -1.0 * cfdata->locals.button; else if (cfdata->locals.delay < 0) cfdata->locals.delay = 0; if (cfdata->locals.add) { EINA_LIST_FOREACH(cfdata->binding.edge, l, bi) if ((bi->modifiers == cfdata->locals.modifiers) && (bi->edge == cfdata->locals.edge) && (bi->drag_only == cfdata->locals.drag_only) && (dblequal(bi->delay * 1000, cfdata->locals.delay * 1000))) { found = 1; break; } } else { if (cfdata->locals.cur && cfdata->locals.cur[0] && (sscanf(cfdata->locals.cur, "e%d", &n) == 1)) { bi = eina_list_nth(cfdata->binding.edge, n); EINA_LIST_FOREACH(cfdata->binding.edge, l, bi2) { if (bi == bi2) continue; if ((bi->modifiers == cfdata->locals.modifiers) && (bi->edge == cfdata->locals.edge) && (bi->drag_only == cfdata->locals.drag_only) && (dblequal(bi->delay * 1000, cfdata->locals.delay * 1000))) { found = 1; break; } } } } if (!found) { if (cfdata->locals.add) { bi = E_NEW(E_Config_Binding_Edge, 1); bi->context = E_BINDING_CONTEXT_ZONE; bi->edge = cfdata->locals.edge; bi->any_mod = 0; bi->delay = (float)(cfdata->locals.delay); bi->action = NULL; bi->params = NULL; bi->modifiers = cfdata->locals.modifiers; bi->drag_only = cfdata->locals.drag_only; cfdata->binding.edge = eina_list_append(cfdata->binding.edge, bi); } else { if (cfdata->locals.cur && cfdata->locals.cur[0] && (sscanf(cfdata->locals.cur, "e%d", &n) == 1)) { bi = eina_list_nth(cfdata->binding.edge, n); bi->modifiers = cfdata->locals.modifiers; bi->delay = cfdata->locals.delay; bi->edge = cfdata->locals.edge; bi->drag_only = cfdata->locals.drag_only; } } if (cfdata->locals.add) { E_Config_Binding_Edge *tmp; char buf[256]; n = 0; _update_edge_binding_list(cfdata); EINA_LIST_FOREACH(cfdata->binding.edge, l, tmp) { if (tmp == bi) break; n++; } e_widget_ilist_selected_set(cfdata->gui.o_binding_list, n); e_widget_ilist_unselect(cfdata->gui.o_action_list); eina_stringshare_del(cfdata->locals.action); cfdata->locals.action = eina_stringshare_add(""); snprintf(buf, sizeof(buf), "%d", (int)-bi->delay); e_widget_label_text_set(cfdata->gui.o_button, NULL); if ((cfdata->params) && (cfdata->params[0])) { int j, g = -1; _find_edge_binding_action("exec", NULL, &g, NULL, &j); if (j >= 0) { e_widget_ilist_unselect(cfdata->gui.o_action_list); e_widget_ilist_selected_set(cfdata->gui.o_action_list, (j + g + 1)); e_widget_entry_clear(cfdata->gui.o_params); e_widget_entry_text_set(cfdata->gui.o_params, cfdata->params); } } else { e_widget_entry_clear(cfdata->gui.o_params); e_widget_disabled_set(cfdata->gui.o_params, 1); } } else if (bi) { char *label; label = _edge_binding_text_get(bi->edge, bi->delay, bi->modifiers, bi->drag_only); e_widget_ilist_nth_label_set(cfdata->gui.o_binding_list, n, label); free(label); } } else { int g, a, j; const char *label = NULL; E_Action_Group *actg = NULL; E_Action_Description *actd = NULL; if (cfdata->locals.add) _find_edge_binding_action(bi->action, bi->params, &g, &a, &j); else _find_edge_binding_action(bi2->action, bi2->params, &g, &a, &j); actg = eina_list_nth(e_action_groups_get(), g); if (actg) actd = eina_list_nth(actg->acts, a); if (actd) label = _(actd->act_name); e_util_dialog_show(_("Edge Binding Error"), _("The edge binding that you chose is already used by" "%s action." "Please choose another edge to bind."), label ? label : _("Unknown")); } } /********** Helper *********************************/ static void _auto_apply_changes(E_Config_Dialog_Data *cfdata) { int n, g, a, ok; E_Config_Binding_Edge *bi; E_Action_Group *actg; E_Action_Description *actd; if ((!cfdata->locals.cur) || (!cfdata->locals.cur[0]) || (!cfdata->locals.action) || (!cfdata->locals.action[0])) return; if (sscanf(cfdata->locals.cur, "e%d", &n) != 1) return; if (sscanf(cfdata->locals.action, "%d %d", &g, &a) != 2) return; bi = eina_list_nth(cfdata->binding.edge, n); if (!bi) return; actg = eina_list_nth(e_action_groups_get(), g); if (!actg) return; actd = eina_list_nth(actg->acts, a); if (!actd) return; eina_stringshare_del(bi->action); bi->action = NULL; if (actd->act_cmd) bi->action = eina_stringshare_add(actd->act_cmd); eina_stringshare_del(bi->params); bi->params = NULL; if (actd->act_params) bi->params = eina_stringshare_add(actd->act_params); else { ok = 1; if (cfdata->locals.params) { if (!strcmp(cfdata->locals.params, TEXT_NO_PARAMS)) ok = 0; if ((actd->param_example) && (!strcmp(cfdata->locals.params, actd->param_example))) ok = 0; } else ok = 0; if (ok) bi->params = eina_stringshare_add(cfdata->locals.params); } } static void _find_edge_binding_action(const char *action, const char *params, int *g, int *a, int *n) { Eina_List *l, *l2; int gg = -1, aa = -1, nn = -1, found; E_Action_Group *actg; E_Action_Description *actd; if (g) *g = -1; if (a) *a = -1; if (n) *n = -1; found = 0; for (l = e_action_groups_get(), gg = 0, nn = 0; l; l = l->next, gg++) { actg = l->data; for (l2 = actg->acts, aa = 0; l2; l2 = l2->next, aa++) { actd = l2->data; if (!strcmp((!action ? "" : action), (!actd->act_cmd ? "" : actd->act_cmd))) { if (!params || !params[0]) { if ((!actd->act_params) || (!actd->act_params[0])) { if (g) *g = gg; if (a) *a = aa; if (n) *n = nn; return; } else continue; } else { if ((!actd->act_params) || (!actd->act_params[0])) { if (g) *g = gg; if (a) *a = aa; if (n) *n = nn; found = 1; } else { if (!strcmp(params, actd->act_params)) { if (g) *g = gg; if (a) *a = aa; if (n) *n = nn; return; } } } } nn++; } if (found) break; } if (!found) { if (g) *g = -1; if (a) *a = -1; if (n) *n = -1; } } static char * _edge_binding_text_get(E_Zone_Edge edge, float delay, int mod, int drag_only) { Eina_Strbuf *b; char *ret; b = eina_strbuf_new(); if (mod & E_BINDING_MODIFIER_ALT) { if (eina_strbuf_length_get(b)) eina_strbuf_append(b, " + "); eina_strbuf_append(b, _("ALT")); } if (mod & E_BINDING_MODIFIER_SHIFT) { if (eina_strbuf_length_get(b)) eina_strbuf_append(b, " + "); eina_strbuf_append(b, _("SHIFT")); } if (mod & E_BINDING_MODIFIER_WIN) { if (eina_strbuf_length_get(b)) eina_strbuf_append(b, " + "); eina_strbuf_append(b, _("WIN")); } if (edge) { if (eina_strbuf_length_get(b)) eina_strbuf_append(b, " + "); switch (edge) { case E_ZONE_EDGE_LEFT: eina_strbuf_append(b, _("Left Edge")); break; case E_ZONE_EDGE_TOP: eina_strbuf_append(b, _("Top Edge")); break; case E_ZONE_EDGE_RIGHT: eina_strbuf_append(b, _("Right Edge")); break; case E_ZONE_EDGE_BOTTOM: eina_strbuf_append(b, _("Bottom Edge")); break; case E_ZONE_EDGE_TOP_LEFT: eina_strbuf_append(b, _("Top Left Edge")); break; case E_ZONE_EDGE_TOP_RIGHT: eina_strbuf_append(b, _("Top Right Edge")); break; case E_ZONE_EDGE_BOTTOM_RIGHT: eina_strbuf_append(b, _("Bottom Right Edge")); break; case E_ZONE_EDGE_BOTTOM_LEFT: eina_strbuf_append(b, _("Bottom Left Edge")); break; default: break; } } if (!eina_flt_exact(delay, 0.0)) { if (eina_strbuf_length_get(b)) eina_strbuf_append(b, " "); if (dblequal(delay, -1.0)) eina_strbuf_append(b, _("(left clickable)")); else if (delay < -1.0) eina_strbuf_append(b, _("(clickable)")); else eina_strbuf_append_printf(b, "%.2fs", delay); } if (drag_only) { if (eina_strbuf_length_get(b)) eina_strbuf_append(b, " "); eina_strbuf_append(b, _("(drag only)")); } ret = eina_strbuf_string_steal(b); eina_strbuf_free(b); if (ret[0]) return ret; free(ret); return strdup(TEXT_NONE_ACTION_EDGE); }