#include "e_mod_main.h" #define MAX_ITEMS 50 typedef struct _Plugin Plugin; struct _Plugin { Evry_Plugin base; Evry_Selector *selector; }; inline static int _is_action(const Evry_Item *it) { return (it->plugin->name == action_selector); } static int _cb_sort_recent(const void *data1, const void *data2) { const Evry_Item *it1 = data1; const Evry_Item *it2 = data2; if (it1->usage && it2->usage) return (it1->usage > it2->usage ? -1 : 1); if (it1->usage && !it2->usage) return -1; if (it2->usage && !it1->usage) return 1; if (_is_action(it1) || _is_action(it2)) { if (_is_action(it1) && _is_action(it2)) return (it1->priority - it2->priority); else if (_is_action(it1)) return ((it1->plugin->config->priority + it1->priority) - (it2->plugin->config->priority)); else return ((it1->plugin->config->priority) - (it2->plugin->config->priority + it2->priority)); } if (it1->plugin == it2->plugin) return (it1->priority - it2->priority); return strcmp(it1->label, it2->label); return 1; } static int _cb_sort(const void *data1, const void *data2) { const Evry_Item *it1 = data1; const Evry_Item *it2 = data2; if (it1->usage && it2->usage) return (it1->usage > it2->usage ? -1 : 1); if (it1->usage && !it2->usage) return -1; if (it2->usage && !it1->usage) return 1; if (_is_action(it1) || _is_action(it2)) { if (_is_action(it1) && _is_action(it2)) return (it1->priority - it2->priority); else if (_is_action(it1)) return ((it1->plugin->config->priority + it1->priority) - it2->plugin->config->priority); else return (it1->plugin->config->priority - (it1->plugin->config->priority + it2->priority)); } if ((it1->plugin == it2->plugin) && (it1->priority - it2->priority)) return (it1->priority - it2->priority); if (it1->fuzzy_match > 0 || it2->fuzzy_match > 0) { if (it2->fuzzy_match <= 0) return -1; if (it1->fuzzy_match <= 0) return 1; if (it1->fuzzy_match - it2->fuzzy_match) return (it1->fuzzy_match - it2->fuzzy_match); } if (it1->plugin->config->priority - it2->plugin->config->priority) return (it1->plugin->config->priority - it2->plugin->config->priority); return strcasecmp(it1->label, it2->label); } static inline Eina_List * _add_item(Plugin *p, Eina_List *items, Evry_Item *it) { /* remove duplicates provided by different plugins */ if (it->id) { Eina_List *_l; Evry_Item *_it; EINA_LIST_FOREACH(p->base.items, _l, _it) { if ((it->plugin->name != _it->plugin->name) && (it->plugin->type_out == _it->plugin->type_out) && (it->id == _it->id)) return items; } } evry_item_ref(it); items = eina_list_append(items, it); EVRY_PLUGIN_ITEM_APPEND(p, it); return items; } static int _fetch(Evry_Plugin *plugin, const char *input) { Plugin *p = (Plugin *) plugin; Evry_Plugin *pp; Evry_State *s; Eina_List *l, *ll, *lp = NULL; Evry_Item *it; int i, cnt = 0; Eina_List *items = NULL; const char *context = NULL; if (input && !input[0]) input = NULL; plugin->changed = 1; EVRY_PLUGIN_ITEMS_FREE(p); s = p->selector->state; if (!s || !s->cur_plugins) return 0; /* get current 'context' ... */ for (i = 1; i < 3; i++) { if (p->selector == selectors[i]) { it = selectors[i-1]->state->cur_item; if (it) context = it->context; } } /* filter all to be shown in aggregator */ EINA_LIST_FOREACH(s->cur_plugins, l, pp) { if (!pp->aggregate || pp == plugin) continue; lp = eina_list_append(lp, pp); } if (!lp) return 0; /* if there is only one plugin append all items */ if (!lp->next) { pp = lp->data; EINA_LIST_FOREACH(pp->items, l, it) { evry_history_item_usage_set(p->selector->history, it, input, context); it->fuzzy_match = evry_fuzzy_match(it->label, input); items = _add_item(p, items, it); } } /* if there is input append all items that match or have fuzzy_match set to -1 */ else if (input) { EINA_LIST_FOREACH(lp, l, pp) { EINA_LIST_FOREACH(pp->items, ll, it) { if (it->fuzzy_match == 0) it->fuzzy_match = evry_fuzzy_match(it->label, input); if (it->fuzzy_match || p->selector == selectors[2]) { evry_history_item_usage_set(p->selector->history, it, input, context); items = _add_item(p, items, it); } } } } /* always append items of action or object selector */ else if ((!input) && ((p->selector == selectors[1]) || (p->selector == selectors[2]))) { EINA_LIST_FOREACH(lp, l, pp) { EINA_LIST_FOREACH(pp->items, ll, it) { evry_history_item_usage_set(p->selector->history, it, NULL, context); it->fuzzy_match = 0; items = _add_item(p, items, it); } } } /* no input: append all items that are in history */ else { EINA_LIST_FOREACH(lp, l, pp) { EINA_LIST_FOREACH(pp->items, ll, it) { if (evry_history_item_usage_set(p->selector->history, it, input, context) && (!eina_list_data_find_list(items, it))) { items = _add_item(p, items, it); } } } } if (eina_list_count(items) < MAX_ITEMS) { EINA_LIST_FOREACH(lp, l, pp) { EINA_LIST_FOREACH(pp->items, ll, it) { if (!eina_list_data_find_list(items, it)) items = _add_item(p, items, it); } } } if (items) eina_list_free(items); if (input) { EVRY_PLUGIN_ITEMS_SORT(p, _cb_sort); } else { EVRY_PLUGIN_ITEMS_SORT(p, _cb_sort_recent); } cnt = 0; EINA_LIST_FOREACH_SAFE(p->base.items, l, ll, it) { if (cnt++ < MAX_ITEMS) continue; evry_item_free(it); p->base.items = eina_list_remove_list(p->base.items, l); } return 1; } static int _action(Evry_Plugin *plugin, const Evry_Item *act, const Evry_Item *subj) { if (act->plugin && act->plugin->action) return act->plugin->action(act->plugin, act, subj); return 0; } static void _cleanup(Evry_Plugin *plugin) { Evry_Item *it; EINA_LIST_FREE(plugin->items, it) evry_item_free(it); } static Evas_Object * _icon_get(Evry_Plugin *plugin, const Evry_Item *it, Evas *e) { Evas_Object *o = NULL; if (it->plugin) { if (it->plugin->icon_get) o = it->plugin->icon_get(it->plugin, it, e); else if (it->plugin->icon) o = evry_icon_theme_get(it->plugin->icon, e); } return o; } Evry_Plugin * evry_plug_aggregator_new(Evry_Selector *sel, int type) { Evry_Plugin *p; p = EVRY_PLUGIN_NEW(Plugin, N_("All"), type, "", "", NULL, _cleanup, _fetch, _icon_get, NULL); p->action = &_action; p->history = EINA_FALSE; evry_plugin_register(p, -1); PLUGIN(pa, p); pa->selector = sel; return p; }