/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ #include "shell-app-system.h" #include #define GMENU_I_KNOW_THIS_IS_UNSTABLE #include enum { PROP_0, }; enum { CHANGED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; struct _ShellAppSystemPrivate { GMenuTree *tree; GMenuTreeDirectory *trunk; GList *cached_menus; }; static void shell_app_system_finalize (GObject *object); static void on_tree_changed (GMenuTree *tree, gpointer user_data); static void reread_menus (ShellAppSystem *self); G_DEFINE_TYPE(ShellAppSystem, shell_app_system, G_TYPE_OBJECT); static gpointer shell_app_menu_entry_copy (gpointer entryp) { ShellAppMenuEntry *entry; ShellAppMenuEntry *copy; entry = entryp; copy = g_new0 (ShellAppMenuEntry, 1); copy->name = g_strdup (entry->name); copy->id = g_strdup (entry->id); copy->icon = g_strdup (entry->icon); return copy; } static void shell_app_menu_entry_free (gpointer entryp) { ShellAppMenuEntry *entry = entryp; g_free (entry->name); g_free (entry->id); g_free (entry->icon); g_free (entry); } static void shell_app_system_class_init(ShellAppSystemClass *klass) { GObjectClass *gobject_class = (GObjectClass *)klass; gobject_class->finalize = shell_app_system_finalize; signals[CHANGED] = g_signal_new ("changed", SHELL_TYPE_APP_SYSTEM, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ShellAppSystemClass, changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); g_type_class_add_private (gobject_class, sizeof (ShellAppSystemPrivate)); } static void shell_app_system_init (ShellAppSystem *self) { ShellAppSystemPrivate *priv; self->priv = priv = G_TYPE_INSTANCE_GET_PRIVATE (self, SHELL_TYPE_APP_SYSTEM, ShellAppSystemPrivate); priv->tree = gmenu_tree_lookup ("applications.menu", GMENU_TREE_FLAGS_NONE); priv->trunk = gmenu_tree_get_root_directory (priv->tree); gmenu_tree_add_monitor (priv->tree, on_tree_changed, self); reread_menus (self); } static void shell_app_system_finalize (GObject *object) { ShellAppSystem *self = SHELL_APP_SYSTEM (object); ShellAppSystemPrivate *priv = self->priv; gmenu_tree_remove_monitor (priv->tree, on_tree_changed, self); gmenu_tree_unref (priv->tree); G_OBJECT_CLASS (shell_app_system_parent_class)->finalize(object); } static void reread_menus (ShellAppSystem *self) { GSList *entries = gmenu_tree_directory_get_contents (self->priv->trunk); GSList *iter; ShellAppSystemPrivate *priv = self->priv; g_list_foreach (self->priv->cached_menus, (GFunc)shell_app_menu_entry_free, NULL); g_list_free (self->priv->cached_menus); self->priv->cached_menus = NULL; for (iter = entries; iter; iter = iter->next) { GMenuTreeItem *item = iter->data; switch (gmenu_tree_item_get_type (item)) { case GMENU_TREE_ITEM_DIRECTORY: { GMenuTreeDirectory *dir = iter->data; ShellAppMenuEntry *shell_entry = g_new0 (ShellAppMenuEntry, 1); shell_entry->name = g_strdup (gmenu_tree_directory_get_name (dir)); shell_entry->id = g_strdup (gmenu_tree_directory_get_menu_id (dir)); shell_entry->icon = g_strdup (gmenu_tree_directory_get_icon (dir)); priv->cached_menus = g_list_prepend (priv->cached_menus, shell_entry); gmenu_tree_item_unref (dir); } break; default: break; } } priv->cached_menus = g_list_reverse (priv->cached_menus); g_slist_free (entries); } static void on_tree_changed (GMenuTree *monitor, gpointer user_data) { ShellAppSystem *self = SHELL_APP_SYSTEM (user_data); g_signal_emit (self, signals[CHANGED], 0); reread_menus (self); } GType shell_app_menu_entry_get_type (void) { static GType gtype = G_TYPE_INVALID; if (gtype == G_TYPE_INVALID) { gtype = g_boxed_type_register_static ("ShellAppMenuEntry", shell_app_menu_entry_copy, shell_app_menu_entry_free); } return gtype; } static GList * gather_entries_recurse (ShellAppSystem *monitor, GList *ids, GMenuTreeDirectory *root) { GSList *contents; GSList *iter; contents = gmenu_tree_directory_get_contents (root); for (iter = contents; iter; iter = iter->next) { GMenuTreeItem *item = iter->data; switch (gmenu_tree_item_get_type (item)) { case GMENU_TREE_ITEM_ENTRY: { GMenuTreeEntry *entry = (GMenuTreeEntry *)item; const char *id = gmenu_tree_entry_get_desktop_file_id (entry); ids = g_list_prepend (ids, g_strdup (id)); } break; case GMENU_TREE_ITEM_DIRECTORY: { GMenuTreeDirectory *dir = (GMenuTreeDirectory*)item; ids = gather_entries_recurse (monitor, ids, dir); } break; default: break; } gmenu_tree_item_unref (item); } g_slist_free (contents); return ids; } /** * shell_app_system_get_applications_for_menu: * * Return value: (transfer full) (element-type utf8): List of desktop file ids */ GList * shell_app_system_get_applications_for_menu (ShellAppSystem *monitor, const char *menu) { char *path; GMenuTreeDirectory *menu_entry; path = g_strdup_printf ("/%s", menu); menu_entry = gmenu_tree_get_directory_from_path (monitor->priv->tree, path); g_free (path); g_assert (menu_entry != NULL); return gather_entries_recurse (monitor, NULL, menu_entry); } /** * shell_app_system_get_menus: * * Return value: (transfer none) (element-type AppMenuEntry): List of toplevel menus */ GList * shell_app_system_get_menus (ShellAppSystem *monitor) { return monitor->priv->cached_menus; }