From c2e94e1925f3c7acfa3fd3352dc8fdafe6375adc Mon Sep 17 00:00:00 2001
From: zhuyaliang <15132211195@163.com>
Date: Wed, 25 Oct 2023 21:43:03 +0800
Subject: Remove libslab library libslab is only used in mate-c-c and there is
 no need to provide a library

---
 shell/app-shell.c | 1448 +++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 1448 insertions(+)
 create mode 100644 shell/app-shell.c

(limited to 'shell/app-shell.c')

diff --git a/shell/app-shell.c b/shell/app-shell.c
new file mode 100644
index 00000000..0e681c2e
--- /dev/null
+++ b/shell/app-shell.c
@@ -0,0 +1,1448 @@
+/*
+ * This file is part of libslab.
+ *
+ * Copyright (c) 2006 Novell, Inc.
+ *
+ * Libslab is free software; you can redistribute it and/or modify it under the
+ * terms of the GNU Lesser General Public License as published by the Free
+ * Software Foundation; either version 2 of the License, or (at your option)
+ * any later version.
+ *
+ * Libslab is distributed in the hope that it will be useful, but WITHOUT ANY
+ * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+ * FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License for
+ * more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with libslab; if not, write to the Free Software Foundation, Inc., 51
+ * Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <libmate-desktop/mate-desktop-item.h>
+#include <gio/gio.h>
+#include <gtk/gtk.h>
+#include <gdk/gdkkeysyms.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <string.h>
+#include <stdlib.h>
+
+#include <glib/gi18n-lib.h>
+
+#include "app-shell.h"
+#include "shell-window.h"
+#include "app-resizer.h"
+#include "slab-section.h"
+#include "slab-mate-util.h"
+#include "search-bar.h"
+
+#include "application-tile.h"
+#include "themed-icon.h"
+
+#define TILE_EXEC_NAME "Tile_desktop_exec_name"
+#define CC_SCHEMA "org.mate.control-center"
+#define EXIT_SHELL_ON_ACTION_START "cc-exit-shell-on-action-start"
+#define EXIT_SHELL_ON_ACTION_HELP "cc-exit-shell-on-action-help"
+#define EXIT_SHELL_ON_ACTION_ADD_REMOVE "cc-exit-shell-on-action-add-remove"
+#define EXIT_SHELL_ON_ACTION_UPGRADE_UNINSTALL "cc-exit-shell-on-action-upgrade-uninstall"
+
+static void create_application_category_sections (AppShellData * app_data);
+static GtkWidget *create_filter_section (AppShellData * app_data, const gchar * title);
+static GtkWidget *create_groups_section (AppShellData * app_data, const gchar * title);
+static GtkWidget *create_actions_section (AppShellData * app_data, const gchar * title,
+	void (*actions_handler) (Tile *, TileEvent *, gpointer));
+
+static void generate_category (const char * category, MateMenuTreeDirectory * root_dir, AppShellData * app_data, gboolean recursive);
+static void generate_launchers (MateMenuTreeDirectory * root_dir, AppShellData * app_data,
+	CategoryData * cat_data, gboolean recursive);
+static void generate_new_apps (AppShellData * app_data);
+static void insert_launcher_into_category (CategoryData * cat_data, MateDesktopItem * desktop_item,
+	AppShellData * app_data);
+
+static gboolean main_keypress_callback (GtkWidget * widget, GdkEventKey * event,
+	AppShellData * app_data);
+static gboolean main_delete_callback (GtkWidget * widget, GdkEvent * event,
+	AppShellData * app_data);
+static void application_launcher_clear_search_bar (AppShellData * app_data);
+static void launch_selected_app (AppShellData * app_data);
+static void generate_potential_apps (gpointer catdata, gpointer user_data);
+
+static void relayout_shell (AppShellData * app_data);
+static gboolean handle_filter_changed (NldSearchBar * search_bar, const char *text,
+	gpointer user_data);
+static void handle_group_clicked (Tile * tile, TileEvent * event, gpointer user_data);
+static void set_state (AppShellData * app_data, GtkWidget * widget);
+static void populate_groups_section (AppShellData * app_data);
+static void generate_filtered_lists (gpointer catdata, gpointer user_data);
+static void show_no_results_message (AppShellData * app_data, GtkWidget * containing_vbox);
+static void populate_application_category_sections (AppShellData * app_data,
+	GtkWidget * containing_vbox);
+static void populate_application_category_section (AppShellData * app_data, SlabSection * section,
+	GList * launcher_list);
+static void tile_activated_cb (Tile * tile, TileEvent * event, gpointer user_data);
+static void handle_launcher_single_clicked (Tile * launcher, gpointer data);
+static void handle_menu_action_performed (Tile * launcher, TileEvent * event, TileAction * action,
+	gpointer data);
+static gint application_launcher_compare (gconstpointer a, gconstpointer b);
+static void matemenu_tree_changed_callback (MateMenuTree * tree, gpointer user_data);
+gboolean regenerate_categories (AppShellData * app_data);
+
+void
+hide_shell (AppShellData * app_data)
+{
+	gtk_window_get_position (GTK_WINDOW (app_data->main_app),
+		&app_data->main_app_window_x, &app_data->main_app_window_y);
+	/* clear the search bar now so reshowing is fast and flicker free - BNC#283186 */
+	application_launcher_clear_search_bar (app_data);
+	gtk_widget_hide (app_data->main_app);
+}
+
+void
+show_shell (AppShellData * app_data)
+{
+	gtk_widget_show_all (app_data->main_app);
+	if (!app_data->static_actions)
+		gtk_widget_hide (app_data->actions_section);  /* don't show unless a launcher is selected */
+
+	if (app_data->main_app_window_shown_once)
+		gtk_window_move (GTK_WINDOW (app_data->main_app),
+			app_data->main_app_window_x, app_data->main_app_window_y);
+
+	/* if this is the first time shown, need to clear this handler */
+	else
+		shell_window_clear_resize_handler (SHELL_WINDOW (app_data->shell));
+	app_data->main_app_window_shown_once = TRUE;
+}
+
+gboolean
+create_main_window (AppShellData * app_data, const gchar * app_name, const gchar * title,
+	const gchar * window_icon, gint width, gint height, gboolean hidden)
+{
+	GtkWidget *main_app = gtk_window_new (GTK_WINDOW_TOPLEVEL);
+	app_data->main_app = main_app;
+	gtk_widget_set_name (main_app, app_name);
+	gtk_window_set_title (GTK_WINDOW (main_app), title);
+	gtk_window_set_default_size(GTK_WINDOW(main_app), width, height);
+	gtk_window_set_icon_name (GTK_WINDOW (main_app), window_icon);
+	gtk_container_add (GTK_CONTAINER (main_app), app_data->shell);
+
+	g_signal_connect (main_app, "delete-event", G_CALLBACK (main_delete_callback), app_data);
+	g_signal_connect (main_app, "key-press-event", G_CALLBACK (main_keypress_callback),
+		app_data);
+
+	gtk_window_set_position (GTK_WINDOW (app_data->main_app), GTK_WIN_POS_CENTER);
+	if (!hidden)
+		show_shell (app_data);
+
+	return TRUE;
+}
+
+static void
+generate_potential_apps (gpointer catdata, gpointer user_data)
+{
+	GHashTable *app_hash = (GHashTable *) user_data;
+	CategoryData *data = (CategoryData *) catdata;
+	gchar *uri;
+
+	GList *launcher_list = data->filtered_launcher_list;
+
+	while (launcher_list)
+	{
+		g_object_get (launcher_list->data, "tile-uri", &uri, NULL);
+		/* eliminate dups of same app in multiple categories */
+		if (!g_hash_table_lookup (app_hash, uri))
+			g_hash_table_insert (app_hash, uri, launcher_list->data);
+		else
+			g_free (uri);
+		launcher_list = g_list_next (launcher_list);
+	}
+}
+
+static gboolean
+return_first_entry (gpointer key, gpointer value, gpointer unused)
+{
+	return TRUE;	/*better way to pull an entry out ? */
+}
+
+static void
+launch_selected_app (AppShellData * app_data)
+{
+	GHashTable *app_hash = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
+	guint num_apps;
+
+	g_list_foreach (app_data->categories_list, generate_potential_apps, app_hash);
+	num_apps = g_hash_table_size (app_hash);
+	if (num_apps == 1)
+	{
+		ApplicationTile *launcher =
+			APPLICATION_TILE (g_hash_table_find (app_hash, return_first_entry, NULL));
+		g_hash_table_destroy (app_hash);
+		handle_launcher_single_clicked (TILE (launcher), app_data);
+		return;
+	}
+
+	g_hash_table_destroy (app_hash);
+}
+
+static gboolean
+main_keypress_callback (GtkWidget * widget, GdkEventKey * event, AppShellData * app_data)
+{
+	GApplication *app;
+
+	if (event->keyval == GDK_KEY_Return)
+	{
+		SlabSection *section = SLAB_SECTION (app_data->filter_section);
+		NldSearchBar *search_bar;
+
+		/* Make sure our implementation has not changed */
+		g_assert (NLD_IS_SEARCH_BAR (section->contents));
+		search_bar = NLD_SEARCH_BAR (section->contents);
+		if (nld_search_bar_has_focus (search_bar))
+		{
+			launch_selected_app (app_data);
+			return TRUE;
+		}
+	}
+
+	/* quit on ESC or Ctl-W or Ctl-Q */
+	if (event->keyval == GDK_KEY_Escape ||
+		((event->keyval == GDK_KEY_w || event->keyval == GDK_KEY_W)	&& (event->state & GDK_CONTROL_MASK)) ||
+		((event->keyval == GDK_KEY_q || event->keyval == GDK_KEY_Q) && (event->state & GDK_CONTROL_MASK)))
+	{
+		if (app_data->exit_on_close)
+		{
+			app=g_application_get_default();
+			g_application_quit(app);
+		}
+		else
+			hide_shell (app_data);
+
+		return TRUE;
+	}
+	return FALSE;
+}
+
+static gboolean
+main_delete_callback (GtkWidget * widget, GdkEvent * event, AppShellData * app_data)
+{
+	GApplication *app;
+
+	if (app_data->exit_on_close)
+	{
+		app=g_application_get_default();
+		g_application_quit(app);
+		return FALSE;
+	}
+
+	hide_shell (app_data);
+	return TRUE;		/* stop the processing of this event */
+}
+
+void
+layout_shell (AppShellData * app_data, const gchar * filter_title, const gchar * groups_title,
+	const gchar * actions_title, GSList * actions,
+	void (*actions_handler) (Tile *, TileEvent *, gpointer))
+{
+	GtkWidget *filter_section;
+	GtkWidget *groups_section;
+	GtkWidget *actions_section;
+
+	GtkWidget *left_vbox;
+	GtkWidget *right_vbox;
+	gint num_cols;
+
+	GtkWidget *sw;
+	GtkAdjustment *adjustment;
+
+	app_data->shell = shell_window_new (app_data);
+	app_data->static_actions = actions;
+
+	right_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
+
+	num_cols = SIZING_SCREEN_WIDTH_LARGE_NUMCOLS;
+
+	GdkRectangle geometry = {0};
+
+	GdkDisplay *display;
+	GdkMonitor *monitor;
+
+	display= gdk_screen_get_display (gdk_screen_get_default ());
+	monitor = gdk_display_get_monitor (display, 0);
+	gdk_monitor_get_geometry (monitor, &geometry);
+
+	if (geometry.width <= SIZING_SCREEN_WIDTH_LARGE)
+	{
+		if (geometry.width <= SIZING_SCREEN_WIDTH_MEDIUM)
+			num_cols = SIZING_SCREEN_WIDTH_SMALL_NUMCOLS;
+		else
+			num_cols = SIZING_SCREEN_WIDTH_MEDIUM_NUMCOLS;
+	}
+	app_data->category_layout =
+		app_resizer_new (GTK_BOX (right_vbox), num_cols, TRUE, app_data);
+
+	sw = gtk_scrolled_window_new (NULL, NULL);
+	gtk_widget_set_vexpand (GTK_WIDGET (sw), TRUE);
+	gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw), GTK_POLICY_AUTOMATIC,
+		GTK_POLICY_AUTOMATIC);
+	gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw), GTK_SHADOW_IN);
+	gtk_container_add (GTK_CONTAINER (sw), app_data->category_layout);
+	adjustment = gtk_scrolled_window_get_vadjustment (GTK_SCROLLED_WINDOW (sw));
+	g_object_set (adjustment, "step-increment", (double) 20, NULL);
+
+	create_application_category_sections (app_data);
+	populate_application_category_sections (app_data, right_vbox);
+	app_resizer_set_table_cache (APP_RESIZER (app_data->category_layout),
+		app_data->cached_tables_list);
+
+	gtk_container_set_focus_vadjustment (GTK_CONTAINER (right_vbox),
+		gtk_scrolled_window_get_vadjustment (GTK_SCROLLED_WINDOW (sw)));
+
+	left_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 15);
+
+	filter_section = create_filter_section (app_data, filter_title);
+	app_data->filter_section = filter_section;
+	gtk_box_pack_start (GTK_BOX (left_vbox), filter_section, FALSE, FALSE, 0);
+
+	groups_section = create_groups_section (app_data, groups_title);
+	app_data->groups_section = groups_section;
+	populate_groups_section (app_data);
+	gtk_box_pack_start (GTK_BOX (left_vbox), groups_section, FALSE, FALSE, 0);
+
+	actions_section = create_actions_section (app_data, actions_title, actions_handler);
+	app_data->actions_section = actions_section;
+	gtk_box_pack_start (GTK_BOX (left_vbox), actions_section, FALSE, FALSE, 0);
+
+	shell_window_set_contents (SHELL_WINDOW (app_data->shell), left_vbox, sw);
+}
+
+static gboolean
+relayout_shell_partial (gpointer user_data)
+{
+	AppShellData *app_data = (AppShellData *) user_data;
+	GtkBox *vbox = APP_RESIZER (app_data->category_layout)->child;
+	CategoryData *data;
+
+	if (app_data->stop_incremental_relayout)
+		return FALSE;
+
+	if (app_data->incremental_relayout_cat_list != NULL)
+	{
+		/* There are still categories to layout */
+		data = (CategoryData *) app_data->incremental_relayout_cat_list->data;
+		if (data->filtered_launcher_list != NULL)
+		{
+			populate_application_category_section (app_data, data->section,
+				data->filtered_launcher_list);
+			gtk_box_pack_start (vbox, GTK_WIDGET (data->section), TRUE, TRUE,
+				0);
+			app_data->filtered_out_everything = FALSE;
+		}
+
+		app_data->incremental_relayout_cat_list =
+			g_list_next (app_data->incremental_relayout_cat_list);
+		return TRUE;
+	}
+
+	/* We're done laying out the categories; finish up */
+	if (app_data->filtered_out_everything)
+		show_no_results_message (app_data, GTK_WIDGET (vbox));
+
+	app_resizer_set_table_cache (APP_RESIZER (app_data->category_layout),
+		app_data->cached_tables_list);
+	populate_groups_section (app_data);
+
+	gtk_widget_show_all (app_data->category_layout);
+	gdk_window_set_cursor (gtk_widget_get_window (app_data->shell), NULL);
+
+	app_data->stop_incremental_relayout = TRUE;
+	return FALSE;
+}
+
+static void
+relayout_shell_incremental (AppShellData * app_data)
+{
+	GtkBox *vbox = APP_RESIZER (app_data->category_layout)->child;
+
+	app_data->stop_incremental_relayout = FALSE;
+	app_data->filtered_out_everything = TRUE;
+	app_data->incremental_relayout_cat_list = app_data->categories_list;
+
+	if (app_data->cached_tables_list)
+		g_list_free (app_data->cached_tables_list);
+	app_data->cached_tables_list = NULL;
+
+	remove_container_entries (GTK_CONTAINER (vbox));
+
+	g_idle_add ((GSourceFunc) relayout_shell_partial, app_data);
+}
+
+static void
+relayout_shell (AppShellData * app_data)
+{
+	GtkWidget *shell = app_data->shell;
+	GtkBox *vbox = APP_RESIZER (app_data->category_layout)->child;
+
+	populate_application_category_sections (app_data, GTK_WIDGET (vbox));
+	app_resizer_set_table_cache (APP_RESIZER (app_data->category_layout),
+		app_data->cached_tables_list);
+	populate_groups_section (app_data);
+
+	gtk_widget_show_all (shell);
+	if (!app_data->static_actions && !app_data->last_clicked_launcher)
+		gtk_widget_hide (app_data->actions_section);  /* don't show unless a launcher is selected */
+}
+
+static GtkWidget *
+create_actions_section (AppShellData * app_data, const gchar * title,
+	void (*actions_handler) (Tile *, TileEvent *, gpointer))
+{
+	GtkWidget *section, *launcher;
+	GtkWidget *vbox;
+	GSList *actions;
+	AppAction *action;
+	AtkObject *a11y_cat;
+
+	g_assert (app_data != NULL);
+
+	section = slab_section_new (title, Style1);
+	g_object_ref (section);
+
+	vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
+	slab_section_set_contents (SLAB_SECTION (section), vbox);
+
+	if (app_data->static_actions)
+	{
+		for (actions = app_data->static_actions; actions; actions = actions->next)
+		{
+			GtkWidget *header;
+
+			action = (AppAction *) actions->data;
+			header = gtk_label_new (action->name);
+			gtk_label_set_line_wrap (GTK_LABEL (header), TRUE);
+			gtk_label_set_max_width_chars (GTK_LABEL (header), 0);
+			gtk_label_set_xalign (GTK_LABEL (header), 0.0);
+			launcher = nameplate_tile_new (NULL, NULL, header, NULL);
+
+			g_object_set_data (G_OBJECT (launcher), APP_ACTION_KEY, action->item);
+			g_signal_connect (launcher, "tile-activated", G_CALLBACK (actions_handler),
+				app_data);
+			gtk_box_pack_start (GTK_BOX (vbox), launcher, FALSE, FALSE, 0);
+
+			a11y_cat = gtk_widget_get_accessible (GTK_WIDGET (launcher));
+			atk_object_set_name (a11y_cat, action->name);
+		}
+	}
+
+	return section;
+}
+
+static GtkWidget *
+create_groups_section (AppShellData * app_data, const gchar * title)
+{
+	GtkWidget *section;
+	GtkWidget *vbox;
+
+	g_assert (app_data != NULL);
+
+	section = slab_section_new (title, Style1);
+	g_object_ref (section);
+
+	vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
+	slab_section_set_contents (SLAB_SECTION (section), vbox);
+
+	return section;
+}
+
+static void
+populate_groups_section (AppShellData * app_data)
+{
+	SlabSection *section = SLAB_SECTION (app_data->groups_section);
+	GtkBox *vbox;
+	GList *cat_list;
+
+	vbox = GTK_BOX (section->contents);
+	remove_container_entries (GTK_CONTAINER (vbox));
+
+	cat_list = app_data->categories_list;
+	do
+	{
+		CategoryData *data = (CategoryData *) cat_list->data;
+		if (NULL != data->filtered_launcher_list)
+		{
+			gtk_widget_set_state_flags (GTK_WIDGET (data->group_launcher), GTK_STATE_FLAG_NORMAL, FALSE);
+			gtk_box_pack_start (vbox, GTK_WIDGET (data->group_launcher),
+				FALSE, FALSE, 0);
+		}
+	}
+	while (NULL != (cat_list = g_list_next (cat_list)));
+}
+
+static void
+handle_group_clicked (Tile * tile, TileEvent * event, gpointer user_data)
+{
+	AppShellData *app_data = (AppShellData *) user_data;
+	GtkWidget *section = NULL;
+	GtkAllocation allocation;
+
+	gint clicked_pos =
+		GPOINTER_TO_INT (g_object_get_data (G_OBJECT (tile), GROUP_POSITION_NUMBER_KEY));
+
+	GList *cat_list = app_data->categories_list;
+
+	gint total = 0;
+	do
+	{
+		CategoryData *cat_data = (CategoryData *) cat_list->data;
+		gint pos =
+			GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cat_data->group_launcher),
+				GROUP_POSITION_NUMBER_KEY));
+		if (pos == clicked_pos)
+		{
+			section = GTK_WIDGET (cat_data->section);
+			break;
+		}
+
+		if (NULL != cat_data->filtered_launcher_list)
+		{
+			gtk_widget_get_allocation (GTK_WIDGET (cat_data->section), &allocation);
+			total += allocation.height;
+		}
+	}
+	while (NULL != (cat_list = g_list_next (cat_list)));
+
+	g_assert (section != NULL);
+	set_state (app_data, section);
+
+	app_resizer_set_vadjustment_value (app_data->category_layout, total);
+}
+
+static void
+set_state (AppShellData * app_data, GtkWidget * widget)
+{
+	if (app_data->selected_group)
+	{
+		slab_section_set_selected (app_data->selected_group, FALSE);
+		app_data->selected_group = NULL;
+	}
+
+	if (widget)
+	{
+		app_data->selected_group = SLAB_SECTION (widget);
+		slab_section_set_selected (SLAB_SECTION (widget), TRUE);
+	}
+	gtk_widget_queue_draw (app_data->shell);
+}
+
+static GtkWidget *
+create_filter_section (AppShellData * app_data, const gchar * title)
+{
+	GtkWidget *section;
+
+	GtkWidget *search_bar;
+
+	section = slab_section_new (title, Style1);
+	g_object_ref (section);
+
+	search_bar = nld_search_bar_new ();
+	nld_search_bar_set_search_timeout (NLD_SEARCH_BAR (search_bar), 0);
+	slab_section_set_contents (SLAB_SECTION (section), search_bar);
+
+	g_signal_connect (search_bar, "search",
+	                  G_CALLBACK (handle_filter_changed),
+	                  app_data);
+
+	return section;
+}
+
+static gboolean
+handle_filter_changed_delayed (gpointer user_data)
+{
+	AppShellData *app_data = (AppShellData *) user_data;
+
+	g_list_foreach (app_data->categories_list, generate_filtered_lists,
+		(gpointer) app_data->filter_string);
+	app_data->last_clicked_launcher = NULL;
+
+	/*  showing the updates incremtally is very visually distracting. Much worse than just blanking until
+	   the incremental work is done and then doing one show. It would be nice to optimize this though
+	   somehow and not even show any change but the cursor change until all the work is done. But since
+	   we do the work incrementally in an idle loop I don't know how else besides hiding to not show
+	   incremental updates
+	 */
+	/* gdk_window_freeze_updates(app_data->category_layout->window); */
+	gtk_widget_hide (app_data->category_layout);
+	app_data->busy_cursor =
+		gdk_cursor_new_for_display (gtk_widget_get_display (app_data->shell), GDK_WATCH);
+	gdk_window_set_cursor (gtk_widget_get_window (app_data->shell), app_data->busy_cursor);
+	g_object_unref (app_data->busy_cursor);
+
+	set_state (app_data, NULL);
+	app_resizer_set_vadjustment_value (app_data->category_layout, 0);
+
+	relayout_shell_incremental (app_data);
+
+	app_data->filter_changed_timeout = 0;
+	return FALSE;
+}
+
+static gboolean
+handle_filter_changed (NldSearchBar * search_bar, const char *text, gpointer data)
+{
+	AppShellData *app_data;
+
+	app_data = (AppShellData *) data;
+
+	if (app_data->filter_string)
+		g_free (app_data->filter_string);
+	app_data->filter_string = g_strdup (text);
+
+	if (app_data->filter_changed_timeout)
+		g_source_remove (app_data->filter_changed_timeout);
+
+	app_data->filter_changed_timeout =
+		g_timeout_add (75, handle_filter_changed_delayed, app_data);
+	app_data->stop_incremental_relayout = TRUE;
+
+	return FALSE;
+}
+
+static void
+generate_filtered_lists (gpointer catdata, gpointer user_data)
+{
+	CategoryData *data = (CategoryData *) catdata;
+
+	/* Fixme - everywhere you use ascii you need to fix up for multibyte */
+	gchar *filter_string = g_ascii_strdown (user_data, -1);
+	gchar *temp1, *temp2;
+	GList *launcher_list = data->launcher_list;
+
+	g_list_free (data->filtered_launcher_list);
+	data->filtered_launcher_list = NULL;
+
+	do
+	{
+		ApplicationTile *launcher = APPLICATION_TILE (launcher_list->data);
+		const gchar *filename;
+
+		temp1 = NULL;
+		temp2 = NULL;
+
+		/* Since the filter may remove this entry from the
+		   container it will not get a mouse out event */
+		gtk_widget_set_state_flags (GTK_WIDGET (launcher), GTK_STATE_FLAG_NORMAL, FALSE);
+		filename = g_object_get_data (G_OBJECT (launcher), TILE_EXEC_NAME); /* do I need to free this */
+
+		temp1 = g_ascii_strdown (launcher->name, -1);
+		if (launcher->description)
+			temp2 = g_ascii_strdown (launcher->description, -1);
+		if (g_strrstr (temp1, filter_string) || (launcher->description
+				&& g_strrstr (temp2, filter_string))
+			|| g_strrstr (filename, filter_string))
+		{
+			data->filtered_launcher_list =
+				g_list_append (data->filtered_launcher_list, launcher);
+		}
+		if (temp1)
+			g_free (temp1);
+		if (temp2)
+			g_free (temp2);
+	}
+	while (NULL != (launcher_list = g_list_next (launcher_list)));
+	g_free (filter_string);
+}
+
+static void
+delete_old_data (AppShellData * app_data)
+{
+	GList *temp;
+	GList *cat_list;
+
+	g_assert (app_data != NULL);
+	g_assert (app_data->categories_list != NULL);
+
+	cat_list = app_data->categories_list;
+
+	do
+	{
+		CategoryData *data = (CategoryData *) cat_list->data;
+		gtk_widget_destroy (GTK_WIDGET (data->section));
+		gtk_widget_destroy (GTK_WIDGET (data->group_launcher));
+		g_object_unref (data->section);
+		g_object_unref (data->group_launcher);
+		g_free (data->category);
+
+		for (temp = data->launcher_list; temp; temp = g_list_next (temp))
+		{
+			g_free (g_object_get_data (G_OBJECT (temp->data), TILE_EXEC_NAME));
+			g_object_unref (temp->data);
+		}
+
+		g_list_free (data->launcher_list);
+		g_list_free (data->filtered_launcher_list);
+		g_free (data);
+	}
+	while (NULL != (cat_list = g_list_next (cat_list)));
+
+	g_list_free (app_data->categories_list);
+	app_data->categories_list = NULL;
+	app_data->selected_group = NULL;
+}
+
+static void
+create_application_category_sections (AppShellData * app_data)
+{
+	GList *cat_list;
+	AtkObject *a11y_cat;
+	gint pos = 0;
+
+	g_assert (app_data != NULL);
+	g_assert (app_data->categories_list != NULL);	/* Fixme - pop up a dialog box and then close */
+
+	cat_list = app_data->categories_list;
+
+	do
+	{
+		CategoryData *data = (CategoryData *) cat_list->data;
+		GtkWidget *header = gtk_label_new (data->category);
+		gchar *markup;
+		GtkWidget *hbox;
+		GtkWidget *table;
+
+		gtk_label_set_xalign (GTK_LABEL (header), 0.0);
+		data->group_launcher = TILE (nameplate_tile_new (NULL, NULL, header, NULL));
+		g_object_ref (data->group_launcher);
+
+		g_object_set_data (G_OBJECT (data->group_launcher), GROUP_POSITION_NUMBER_KEY,
+			GINT_TO_POINTER (pos));
+		pos++;
+		g_signal_connect (data->group_launcher, "tile-activated",
+			G_CALLBACK (handle_group_clicked), app_data);
+		a11y_cat = gtk_widget_get_accessible (GTK_WIDGET (data->group_launcher));
+		atk_object_set_name (a11y_cat, data->category);
+
+		markup = g_markup_printf_escaped ("<span size=\"x-large\" weight=\"bold\">%s</span>",
+			data->category);
+		data->section = SLAB_SECTION (slab_section_new_with_markup (markup, Style2));
+
+		/* as we filter these will be added/removed from parent container and we dont want them destroyed */
+		g_object_ref (data->section);
+		g_free (markup);
+
+		hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
+		table = gtk_grid_new ();
+		gtk_grid_set_column_spacing (GTK_GRID (table), 5);
+		gtk_grid_set_row_spacing (GTK_GRID (table), 5);
+		gtk_box_pack_start (GTK_BOX (hbox), table, FALSE, FALSE, 15);
+		slab_section_set_contents (SLAB_SECTION (data->section), hbox);
+	}
+	while (NULL != (cat_list = g_list_next (cat_list)));
+}
+
+static void
+show_no_results_message (AppShellData * app_data, GtkWidget * containing_vbox)
+{
+	gchar *markup;
+	gchar *str1;
+	gchar *str2;
+
+	if (!app_data->filtered_out_everything_widget)
+	{
+		GtkWidget *image;
+		GtkWidget *label;
+
+		app_data->filtered_out_everything_widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
+		gtk_widget_set_halign (app_data->filtered_out_everything_widget, GTK_ALIGN_CENTER);
+		gtk_widget_set_valign (app_data->filtered_out_everything_widget, GTK_ALIGN_CENTER);
+		g_object_ref (app_data->filtered_out_everything_widget);
+
+		image = themed_icon_new ("face-surprise", GTK_ICON_SIZE_DIALOG);
+		gtk_box_pack_start (GTK_BOX (app_data->filtered_out_everything_widget), image, FALSE, FALSE, 0);
+
+		label = gtk_label_new (NULL);
+		gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
+		gtk_box_pack_start (GTK_BOX (app_data->filtered_out_everything_widget), label, TRUE, TRUE, 15);
+		app_data->filtered_out_everything_widget_label = GTK_LABEL (label);
+	}
+
+	str1 = g_markup_printf_escaped ("<b>%s</b>", app_data->filter_string);
+	str2 = g_strdup_printf (_("Your filter \"%s\" does not match any items."), str1);
+	markup = g_strdup_printf ("<span size=\"large\"><b>%s</b></span>\n\n%s",
+		_("No matches found."), str2);
+	gtk_label_set_text (app_data->filtered_out_everything_widget_label, markup);
+	gtk_label_set_use_markup (app_data->filtered_out_everything_widget_label, TRUE);
+	gtk_box_pack_start (GTK_BOX (containing_vbox), app_data->filtered_out_everything_widget,
+		TRUE, TRUE, 0);
+	g_free (str1);
+	g_free (str2);
+	g_free (markup);
+}
+
+static void
+populate_application_category_sections (AppShellData * app_data, GtkWidget * containing_vbox)
+{
+	GList *cat_list = app_data->categories_list;
+	gboolean filtered_out_everything = TRUE;
+	if (app_data->cached_tables_list)
+		g_list_free (app_data->cached_tables_list);
+	app_data->cached_tables_list = NULL;
+
+	remove_container_entries (GTK_CONTAINER (containing_vbox));
+	do
+	{
+		CategoryData *data = (CategoryData *) cat_list->data;
+		if (NULL != data->filtered_launcher_list)
+		{
+			populate_application_category_section (app_data, data->section,
+				data->filtered_launcher_list);
+			gtk_box_pack_start (GTK_BOX (containing_vbox), GTK_WIDGET (data->section),
+				TRUE, TRUE, 0);
+			filtered_out_everything = FALSE;
+		}
+	}
+	while (NULL != (cat_list = g_list_next (cat_list)));
+
+	if (TRUE == filtered_out_everything)
+		show_no_results_message (app_data, containing_vbox);
+}
+
+static void
+populate_application_category_section (AppShellData * app_data, SlabSection * section,
+	GList * launcher_list)
+{
+	GtkWidget *hbox;
+	GtkGrid *table;
+	GList *children;
+
+	hbox = GTK_WIDGET (section->contents);
+
+	children = gtk_container_get_children (GTK_CONTAINER (hbox));
+	table = children->data;
+	g_list_free (children);
+
+	/* Make sure our implementation has not changed and it's still a GtkGrid */
+	g_assert (GTK_IS_GRID (table));
+
+	app_data->cached_tables_list = g_list_append (app_data->cached_tables_list, table);
+
+	app_resizer_layout_table_default (APP_RESIZER (app_data->category_layout), table,
+		launcher_list);
+
+}
+
+gboolean
+regenerate_categories (AppShellData * app_data)
+{
+	delete_old_data (app_data);
+	generate_categories (app_data);
+	create_application_category_sections (app_data);
+	relayout_shell (app_data);
+
+	return FALSE;	/* remove this function from the list */
+}
+
+static void
+matemenu_tree_changed_callback (MateMenuTree * old_tree, gpointer user_data)
+{
+	/*
+	This method only gets called on the first change (matemenu appears to ignore subsequent) until
+	we reget the root dir which we can't do in this method because if we do for some reason this
+	method then gets called multiple times for one actual change. This actually is okay because
+	it's probably a good idea to wait a couple seconds to regenerate the categories in case there
+	are multiple quick changes being made, no sense regenerating multiple times.
+	*/
+	GError *error = NULL;
+	AppShellData * app_data = user_data;
+	if (!matemenu_tree_load_sync (app_data->tree, &error)) {
+		g_warning ("Menu tree loading got error:%s\n", error->message);
+		g_object_unref (app_data->tree);
+		app_data->tree = NULL;
+		g_error_free (error);
+	} else {
+		g_timeout_add_full (G_PRIORITY_DEFAULT_IDLE, 3000, (GSourceFunc) regenerate_categories,
+		                    user_data, NULL);
+	}
+}
+
+AppShellData *
+appshelldata_new (const gchar * menu_name, GtkIconSize icon_size, gboolean show_tile_generic_name, gboolean exit_on_close, gint new_apps_max_items)
+{
+	AppShellData *app_data = g_new0 (AppShellData, 1);
+	app_data->settings = g_settings_new (CC_SCHEMA);
+	app_data->menu_name = menu_name;
+	app_data->icon_size = icon_size;
+	app_data->stop_incremental_relayout = TRUE;
+	app_data->show_tile_generic_name = show_tile_generic_name;
+	app_data->exit_on_close = exit_on_close;
+	if (new_apps_max_items > 0) {
+		app_data->new_apps = g_new0 (NewAppConfig, 1);
+		app_data->new_apps->max_items = new_apps_max_items;
+		app_data->new_apps->name = _("New Applications");
+	}
+	return app_data;
+}
+
+void
+generate_categories (AppShellData * app_data)
+{
+	MateMenuTreeDirectory *root_dir;
+	gboolean need_misc = FALSE;
+	MateMenuTreeIter *iter;
+	MateMenuTreeItemType type;
+
+	if (!app_data->tree)
+	{
+		GError *error = NULL;
+
+		app_data->tree = matemenu_tree_new (app_data->menu_name, MATEMENU_TREE_FLAGS_NONE);
+		g_signal_connect (app_data->tree, "changed", G_CALLBACK (matemenu_tree_changed_callback), app_data);
+		if (! matemenu_tree_load_sync (app_data->tree, &error)) {
+			g_warning("Menu tree loading got error:%s\n", error->message);
+			g_error_free(error);
+			g_object_unref(app_data->tree);
+			app_data->tree = NULL;
+		}
+	}
+
+	if (app_data->tree != NULL)
+		root_dir = matemenu_tree_get_root_directory (app_data->tree);
+	else
+		root_dir = NULL;
+
+	if ( app_data->tree == NULL || root_dir == NULL) {
+		GtkWidget *dialog = gtk_message_dialog_new (NULL, GTK_DIALOG_DESTROY_WITH_PARENT,
+				GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "Failure loading - %s",
+				app_data->menu_name);
+		gtk_dialog_run (GTK_DIALOG (dialog));
+		gtk_widget_destroy (dialog);
+		exit (1);	/* Fixme - is there a MATE/GTK way to do this. */
+	}
+
+	iter = matemenu_tree_directory_iter (root_dir);
+	while ((type = matemenu_tree_iter_next (iter)) != MATEMENU_TREE_ITEM_INVALID) {
+		gpointer item;
+		const char *category;
+		switch (type) {
+			case MATEMENU_TREE_ITEM_DIRECTORY:
+				item = matemenu_tree_iter_get_directory (iter);
+				category = matemenu_tree_directory_get_name (item);
+				generate_category(category, item, app_data, TRUE);
+				matemenu_tree_item_unref (item);
+				break;
+			case MATEMENU_TREE_ITEM_ENTRY:
+				need_misc = TRUE;
+				break;
+			default:
+				break;
+		}
+	}
+	matemenu_tree_iter_unref(iter);
+
+	if (need_misc)
+		generate_category (_("Other"), root_dir, app_data, FALSE);
+
+	if (app_data->hash)
+	{
+		g_hash_table_destroy (app_data->hash);
+		app_data->hash = NULL;
+	}
+
+	matemenu_tree_item_unref (root_dir);
+
+	if (app_data->new_apps && (app_data->new_apps->max_items > 0))
+		generate_new_apps (app_data);
+}
+
+static void
+generate_category (const char * category, MateMenuTreeDirectory * root_dir, AppShellData * app_data, gboolean recursive)
+{
+	CategoryData *data;
+	/* This is not needed. MateMenu already returns an ordered, non duplicate list
+	GList *list_entry;
+	list_entry =
+		g_list_find_custom (app_data->categories_list, category,
+		category_name_compare);
+	if (!list_entry)
+	{
+	*/
+		data = g_new0 (CategoryData, 1);
+		data->category = g_strdup (category);
+		app_data->categories_list =
+			/* use the matemenu order instead of alphabetical */
+			g_list_append (app_data->categories_list, data);
+			/* g_list_insert_sorted (app_data->categories_list, data, category_data_compare); */
+	/*
+	}
+	else
+	{
+		data = list_entry->data;
+	}
+	*/
+
+	if (app_data->hash)	/* used to eliminate dups on a per category basis. */
+		g_hash_table_destroy (app_data->hash);
+	app_data->hash = g_hash_table_new (g_str_hash, g_str_equal);
+	generate_launchers (root_dir, app_data, data, recursive);
+}
+
+static gboolean
+check_specific_apps_hack (MateDesktopItem * item)
+{
+	static const gchar *COMMAND_LINE_LOCKDOWN_SCHEMA = "org.mate.lockdown";
+	static const gchar *COMMAND_LINE_LOCKDOWN_KEY = "disable-command-line";
+	static const gchar *COMMAND_LINE_LOCKDOWN_DESKTOP_CATEGORY = "TerminalEmulator";
+	static gboolean got_lockdown_value = FALSE;
+	static gboolean command_line_lockdown;
+
+	gchar *path;
+	const char *exec;
+
+	if (!got_lockdown_value)
+	{
+		got_lockdown_value = TRUE;
+		GSettings *lockdown_settings;
+		lockdown_settings = g_settings_new (COMMAND_LINE_LOCKDOWN_SCHEMA);
+		command_line_lockdown = g_settings_get_boolean (lockdown_settings, COMMAND_LINE_LOCKDOWN_KEY);
+		g_object_unref (lockdown_settings);
+	}
+
+	/* This seems like an ugly hack but it's the way it's currently done in the old control center */
+	exec = mate_desktop_item_get_string (item, MATE_DESKTOP_ITEM_EXEC);
+
+	/* discard xscreensaver if mate-screensaver is installed */
+	if ((exec && !strcmp (exec, "xscreensaver-demo"))
+		&& (path = g_find_program_in_path ("mate-screensaver-preferences")))
+	{
+		g_free (path);
+		return TRUE;
+	}
+
+	/* discard gnome-keyring-manager if CASA is installed */
+	if ((exec && !strcmp (exec, "gnome-keyring-manager"))
+		&& (path = g_find_program_in_path ("CASAManager.sh")))
+	{
+		g_free (path);
+		return TRUE;
+	}
+
+	/* discard terminals if lockdown key is set */
+	if (command_line_lockdown)
+	{
+		const gchar *categories =
+			mate_desktop_item_get_string (item, MATE_DESKTOP_ITEM_CATEGORIES);
+		if (g_strrstr (categories, COMMAND_LINE_LOCKDOWN_DESKTOP_CATEGORY))
+		{
+			return TRUE;
+		}
+	}
+
+	return FALSE;
+}
+
+static void
+generate_launchers (MateMenuTreeDirectory * root_dir, AppShellData * app_data, CategoryData * cat_data, gboolean recursive)
+{
+	MateDesktopItem *desktop_item;
+	const gchar *desktop_file;
+	MateMenuTreeIter *iter;
+	MateMenuTreeItemType type;
+
+	iter = matemenu_tree_directory_iter (root_dir);
+	while ((type = matemenu_tree_iter_next (iter)) != MATEMENU_TREE_ITEM_INVALID) {
+		gpointer item;
+		switch (type) {
+			case MATEMENU_TREE_ITEM_DIRECTORY:
+				item = matemenu_tree_iter_get_directory(iter);
+				/* g_message ("Found sub-category %s", matemenu_tree_directory_get_name (item)); */
+				if (recursive)
+					generate_launchers (item, app_data, cat_data, TRUE);
+				matemenu_tree_item_unref (item);
+				break;
+			case MATEMENU_TREE_ITEM_ENTRY:
+				item = matemenu_tree_iter_get_entry(iter);
+				/* g_message ("Found item name is:%s", matemenu_tree_entry_get_desktop_file_id(item)); */
+				desktop_file = matemenu_tree_entry_get_desktop_file_path (item);
+				if (desktop_file)
+				{
+					if (g_hash_table_lookup (app_data->hash, desktop_file))
+					{
+						break;	/* duplicate */
+					}
+					/* Fixme - make sure it's safe to store this without duping it. As far as I can tell it is
+					   safe as long as I don't hang on to this anylonger than I hang on to the MateMenuTreeEntry*
+					   which brings up another point - am I supposed to free these or does freeing the top level recurse
+					   */
+					g_hash_table_insert (app_data->hash, (gpointer) desktop_file,
+							(gpointer) desktop_file);
+				}
+				desktop_item = mate_desktop_item_new_from_file (desktop_file, 0, NULL);
+				if (!desktop_item)
+				{
+					g_critical ("Failure - mate_desktop_item_new_from_file(%s)",
+							desktop_file);
+					break;
+				}
+				if (!check_specific_apps_hack (desktop_item))
+					insert_launcher_into_category (cat_data, desktop_item, app_data);
+				mate_desktop_item_unref (desktop_item);
+				matemenu_tree_item_unref (item);
+				break;
+			default:
+				break;
+		}
+	}
+	matemenu_tree_iter_unref(iter);
+}
+
+static void
+generate_new_apps (AppShellData * app_data)
+{
+	GHashTable *all_apps_cache = NULL;
+	gchar *all_apps;
+	GError *error = NULL;
+	gchar *separator = "\n";
+
+	gchar *all_apps_file_name;
+	gchar **all_apps_split;
+	gint x;
+	gboolean got_new_apps;
+	CategoryData *new_apps_category = NULL;
+	GList *categories, *launchers;
+	GHashTable *new_apps_dups;
+
+	all_apps_file_name = g_build_filename (g_get_user_config_dir (), "mate", "ab-newapps.txt", NULL);
+
+	if (!g_file_get_contents (all_apps_file_name, &all_apps, NULL, &error))
+	{
+		/* If file does not exist, this is the first time this user has run this, create the baseline file */
+		GList *categories, *launchers;
+		GString *gstr;
+		gchar *dirname;
+
+		g_error_free (error);
+		error = NULL;
+
+		/* best initial size determined by running on a couple different platforms */
+		gstr = g_string_sized_new (10000);
+
+		for (categories = app_data->categories_list; categories; categories = categories->next)
+		{
+			CategoryData *data = categories->data;
+			for (launchers = data->launcher_list; launchers; launchers = launchers->next)
+			{
+				Tile *tile = TILE (launchers->data);
+				MateDesktopItem *item =
+					application_tile_get_desktop_item (APPLICATION_TILE (tile));
+				const gchar *uri = mate_desktop_item_get_location (item);
+				g_string_append (gstr, uri);
+				g_string_append (gstr, separator);
+			}
+		}
+
+		dirname = g_path_get_dirname (all_apps_file_name);
+		g_mkdir_with_parents (dirname, 0700);	/* creates if does not exist */
+		g_free (dirname);
+
+		if (!g_file_set_contents (all_apps_file_name, gstr->str, -1, &error))
+			g_warning ("Error setting all apps file:%s\n", error->message);
+
+		g_string_free (gstr, TRUE);
+		g_free (all_apps_file_name);
+		return;
+	}
+
+	all_apps_cache = g_hash_table_new (g_str_hash, g_str_equal);
+	all_apps_split = g_strsplit (all_apps, separator, -1);
+	for (x = 0; all_apps_split[x]; x++)
+	{
+		g_hash_table_insert (all_apps_cache, all_apps_split[x], all_apps_split[x]);
+	}
+
+	got_new_apps = FALSE;
+	new_apps_dups = g_hash_table_new (g_str_hash, g_str_equal);
+	for (categories = app_data->categories_list; categories; categories = categories->next)
+	{
+		CategoryData *cat_data = categories->data;
+		for (launchers = cat_data->launcher_list; launchers; launchers = launchers->next)
+		{
+			Tile *tile = TILE (launchers->data);
+			MateDesktopItem *item =
+				application_tile_get_desktop_item (APPLICATION_TILE (tile));
+			const gchar *uri = mate_desktop_item_get_location (item);
+			if (!g_hash_table_lookup (all_apps_cache, uri))
+			{
+				GFile *file;
+				GFileInfo *info;
+				long filetime;
+
+				if (g_hash_table_lookup (new_apps_dups, uri))
+				{
+					/* if a desktop file is in 2 or more top level categories, only show it once */
+					break;
+				}
+				g_hash_table_insert (new_apps_dups, (gpointer) uri, (gpointer) uri);
+
+				if (!got_new_apps)
+				{
+					new_apps_category = g_new0 (CategoryData, 1);
+					new_apps_category->category =
+						g_strdup (app_data->new_apps->name);
+					app_data->new_apps->garray =
+						g_array_sized_new (FALSE, TRUE,
+						sizeof (NewAppData *),
+						app_data->new_apps->max_items);
+
+					/* should not need this, but a bug in glib does not actually clear the elements until you call this method */
+					g_array_set_size (app_data->new_apps->garray, app_data->new_apps->max_items);
+					got_new_apps = TRUE;
+				}
+
+				file = g_file_new_for_uri (uri);
+				info = g_file_query_info (file,
+							  G_FILE_ATTRIBUTE_TIME_MODIFIED,
+							  0, NULL, NULL);
+
+				if (!info)
+				{
+					g_object_unref (file);
+					g_warning ("Cant get vfs info for %s\n", uri);
+					if (new_apps_category) {
+						g_free (new_apps_category->category);
+						g_free (new_apps_category);
+					}
+					g_free (all_apps_file_name);
+					g_strfreev (all_apps_split);
+					return;
+				}
+				filetime = (long) g_file_info_get_attribute_uint64 (info,
+										    G_FILE_ATTRIBUTE_TIME_MODIFIED);
+				g_object_unref (info);
+				g_object_unref (file);
+
+				for (x = 0; x < app_data->new_apps->max_items; x++)
+				{
+					NewAppData *temp_data = (NewAppData *)
+						g_array_index (app_data->new_apps->garray, NewAppData *, x);
+					if (!temp_data || filetime > temp_data->time)	/* if this slot is empty or we are newer than this slot */
+					{
+						NewAppData *temp = g_new0 (NewAppData, 1);
+						temp->time = filetime;
+						temp->item = item;
+						g_array_insert_val (app_data->new_apps->garray, x,
+							temp);
+						break;
+					}
+				}
+			}
+		}
+	}
+	g_hash_table_destroy (new_apps_dups);
+	g_hash_table_destroy (all_apps_cache);
+
+	if (got_new_apps)
+	{
+		for (x = 0; x < app_data->new_apps->max_items; x++)
+		{
+			NewAppData *data =
+				(NewAppData *) g_array_index (app_data->new_apps->garray,
+				NewAppData *, x);
+			if (data)
+			{
+				insert_launcher_into_category (new_apps_category, data->item,
+					app_data);
+				g_free (data);
+			}
+			else
+				break;
+		}
+		app_data->categories_list =
+			g_list_prepend (app_data->categories_list, new_apps_category);
+
+		g_array_free (app_data->new_apps->garray, TRUE);
+	}
+	g_free (all_apps);
+	g_free (all_apps_file_name);
+	g_strfreev (all_apps_split);
+}
+
+static void
+insert_launcher_into_category (CategoryData * cat_data, MateDesktopItem * desktop_item,
+	AppShellData * app_data)
+{
+	GtkWidget *launcher;
+	static GtkSizeGroup *icon_group = NULL;
+
+	gchar *filepath;
+	gchar *filename;
+	GtkWidget *tile_icon;
+
+	if (!icon_group)
+		icon_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
+
+	launcher =
+		application_tile_new_full (mate_desktop_item_get_location (desktop_item),
+		app_data->icon_size, app_data->show_tile_generic_name);
+	gtk_widget_set_size_request (launcher, SIZING_TILE_WIDTH, -1);
+
+	filepath =
+		g_strdup (mate_desktop_item_get_string (desktop_item, MATE_DESKTOP_ITEM_EXEC));
+	g_strdelimit (filepath, " ", '\0');	/* just want the file name - no args or replacements */
+	filename = g_strrstr (filepath, "/");
+	if (filename)
+		g_stpcpy (filepath, filename + 1);
+	filename = g_ascii_strdown (filepath, -1);
+	g_free (filepath);
+	g_object_set_data (G_OBJECT (launcher), TILE_EXEC_NAME, filename);
+
+	tile_icon = NAMEPLATE_TILE (launcher)->image;
+	gtk_size_group_add_widget (icon_group, tile_icon);
+
+	g_signal_connect (launcher, "tile-activated", G_CALLBACK (tile_activated_cb), app_data);
+
+	/* Note that this will handle the case of the action being launched via the side panel as
+	   well as directly from the context menu of an individual launcher, because they both
+	   funnel through tile_button_action_activate.
+	*/
+	g_signal_connect (launcher, "tile-action-triggered",
+		G_CALLBACK (handle_menu_action_performed), app_data);
+
+	/* These will be inserted/removed from tables as the filter changes and we dont want them */
+	/* destroyed when they are removed */
+	g_object_ref (launcher);
+
+	/* use alphabetical order instead of the matemenu order. We group all sub items in each top level
+	category together, ignoring sub menus, so we also ignore sub menu layout hints */
+	cat_data->launcher_list =
+		/* g_list_insert (cat_data->launcher_list, launcher, -1); */
+		g_list_insert_sorted (cat_data->launcher_list, launcher, application_launcher_compare);
+	cat_data->filtered_launcher_list =
+		/* g_list_insert (cat_data->filtered_launcher_list, launcher, -1); */
+		g_list_insert_sorted (cat_data->filtered_launcher_list, launcher, application_launcher_compare);
+}
+
+static gint
+application_launcher_compare (gconstpointer a, gconstpointer b)
+{
+	ApplicationTile *launcher1 = APPLICATION_TILE (a);
+	ApplicationTile *launcher2 = APPLICATION_TILE (b);
+
+	gchar *val1 = launcher1->name;
+	gchar *val2 = launcher2->name;
+
+	if (val1 == NULL || val2 == NULL)
+	{
+		g_assert_not_reached ();
+	}
+	return g_ascii_strcasecmp (val1, val2);
+}
+
+static void
+application_launcher_clear_search_bar (AppShellData * app_data)
+{
+	SlabSection *section = SLAB_SECTION (app_data->filter_section);
+	NldSearchBar *search_bar;
+	g_assert (NLD_IS_SEARCH_BAR (section->contents));
+	search_bar = NLD_SEARCH_BAR (section->contents);
+	nld_search_bar_set_text (search_bar, "", TRUE);
+}
+
+/*
+static gint
+category_name_compare (gconstpointer a, gconstpointer b)
+{
+	CategoryData *data = (CategoryData *) a;
+	const gchar *category = b;
+
+	if (category == NULL || data->category == NULL)
+	{
+		g_assert_not_reached ();
+	}
+	return g_ascii_strcasecmp (category, data->category);
+}
+*/
+
+static void
+tile_activated_cb (Tile * tile, TileEvent * event, gpointer user_data)
+{
+	switch (event->type)
+	{
+	case TILE_EVENT_ACTIVATED_SINGLE_CLICK:
+	case TILE_EVENT_ACTIVATED_KEYBOARD:
+		handle_launcher_single_clicked (tile, user_data);
+		break;
+	default:
+		break;
+	}
+
+}
+
+static void
+handle_launcher_single_clicked (Tile * launcher, gpointer data)
+{
+	GApplication *app;
+	AppShellData *app_data = (AppShellData *) data;
+
+	tile_trigger_action (launcher, launcher->actions[APPLICATION_TILE_ACTION_START]);
+
+	if (g_settings_get_boolean (app_data->settings, EXIT_SHELL_ON_ACTION_START))
+	{
+		if (app_data->exit_on_close)
+		{
+			app=g_application_get_default();
+			g_application_quit(app);
+		}
+		else
+			hide_shell (app_data);
+	}
+}
+
+static void
+handle_menu_action_performed (Tile * launcher, TileEvent * event, TileAction * action,
+	gpointer data)
+{
+	GApplication *app;
+	AppShellData *app_data = (AppShellData *) data;
+	gchar *temp;
+
+	temp = NULL;
+	if (action == launcher->actions[APPLICATION_TILE_ACTION_START])
+	{
+		temp = EXIT_SHELL_ON_ACTION_START;
+	}
+
+	else if (action == launcher->actions[APPLICATION_TILE_ACTION_HELP])
+	{
+		temp = EXIT_SHELL_ON_ACTION_HELP;
+	}
+
+	else if (action == launcher->actions[APPLICATION_TILE_ACTION_UPDATE_MAIN_MENU]
+		|| action == launcher->actions[APPLICATION_TILE_ACTION_UPDATE_STARTUP])
+	{
+		temp = EXIT_SHELL_ON_ACTION_ADD_REMOVE;
+	}
+
+	else if (action == launcher->actions[APPLICATION_TILE_ACTION_UPGRADE_PACKAGE]
+		|| action == launcher->actions[APPLICATION_TILE_ACTION_UNINSTALL_PACKAGE])
+	{
+		temp = EXIT_SHELL_ON_ACTION_UPGRADE_UNINSTALL;
+	}
+
+	if (temp)
+	{
+		if (g_settings_get_boolean (app_data->settings, temp))
+		{
+			if (app_data->exit_on_close)
+			{
+				app=g_application_get_default();
+				g_application_quit(app);
+			}
+			else
+				hide_shell (app_data);
+		}
+	}
+	else
+		g_warning ("Unknown Action");
+}
-- 
cgit v1.2.1