diff --git a/lib/aspell.c b/lib/aspell.c
deleted file mode 100644
index 8dba66e91ad79fb6eb9cd7201a038f5a013b0d56..0000000000000000000000000000000000000000
--- a/lib/aspell.c
+++ /dev/null
@@ -1,229 +0,0 @@
-/*  $Id$
- *
- *  Copyright 2006-2008 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
- *
- *  This program is free software; you can redistribute it and/or modify
- *  it under the terms of the GNU General Public License as published by
- *  the Free Software Foundation; either version 2 of the License, or
- *  (at your option) any later version.
- *
- *  This program 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 Library General Public License for more details.
- *
- *  You should have received a copy of the GNU General Public License
- *  along with this program; if not, write to the Free Software
- *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-
-
-/* Code to execute the aspell (or ispell or anything command line compatible) binary
- * with a given search term and reads it output. */
-
-
-#ifdef HAVE_CONFIG_H
-#include <config.h>
-#endif
-
-#include <stdlib.h>
-#include <sys/types.h>
-#include <sys/wait.h>
-#include <string.h>
-
-#include <gtk/gtk.h>
-
-#include <libxfcegui4/libxfcegui4.h>
-
-#include "common.h"
-#include "aspell.h"
-#include "gui.h"
-
-
-typedef struct
-{
-	DictData *dd;
-	gchar *word;
-} iodata;
-
-
-static GIOChannel *set_up_io_channel(gint fd, GIOCondition cond, GIOFunc func, gpointer data)
-{
-	GIOChannel *ioc;
-
-	ioc = g_io_channel_unix_new(fd);
-
-	g_io_channel_set_flags(ioc, G_IO_FLAG_NONBLOCK, NULL);
-	g_io_channel_set_encoding(ioc, NULL, NULL);
-	/* "auto-close" */
-	g_io_channel_set_close_on_unref(ioc, TRUE);
-
-	g_io_add_watch(ioc, cond, func, (gpointer) data);
-	g_io_channel_unref(ioc);
-
-	return ioc;
-}
-
-
-static gboolean iofunc_read(GIOChannel *ioc, GIOCondition cond, gpointer data)
-{
-	iodata *iod = data;
-	if (cond & (G_IO_IN | G_IO_PRI))
-	{
-		gchar *msg, *tmp;
-		DictData *dd = iod->dd;
-
-		while (g_io_channel_read_line(ioc, &msg, NULL, NULL, NULL) && msg != NULL)
-		{
-			if (msg[0] == '&')
-			{	/* & cmd 17 7: ... */
-				gint count;
-				tmp = strchr(msg + 2, ' ') + 1;
-				count = atoi(tmp);
-				dict_gui_status_add(dd, ngettext("%d suggestion found.",
-                                            "%d suggestions found.",
-                                            count), count);
-
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n", 1);
-				tmp = g_strdup_printf(_("Suggestions for \"%s\":"), iod->word);
-				gtk_text_buffer_insert_with_tags_by_name(
-					dd->main_textbuffer, &dd->textiter, tmp, -1, "bold", NULL);
-				g_free(tmp);
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n", 1);
-
-				tmp = strchr(msg, ':') + 2;
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, g_strchomp(tmp), -1);
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n\n", 2);
-			}
-			else if (msg[0] == '*')
-			{
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n", 1);
-				tmp = g_strdup_printf(_("\"%s\" is spelled correctly."), iod->word);
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, tmp, -1);
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n\n", 2);
-				g_free(tmp);
-			}
-			else if (msg[0] == '#')
-			{
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n", 1);
-				tmp = g_strdup_printf(_("No suggestions could be found for \"%s\"."),
-					iod->word);
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, tmp, -1);
-				gtk_text_buffer_insert(dd->main_textbuffer, &dd->textiter, "\n\n", 2);
-				g_free(tmp);
-			}
-			g_free(msg);
-		}
-	}
-	if (cond & (G_IO_ERR | G_IO_HUP | G_IO_NVAL))
-	{
-		g_free(iod->word);
-		g_free(iod);
-
-		return FALSE;
-	}
-
-	return TRUE;
-}
-
-
-static gboolean iofunc_read_err(GIOChannel *ioc, GIOCondition cond, gpointer data)
-{
-	if (cond & (G_IO_IN | G_IO_PRI))
-	{
-		gchar *msg;
-		DictData *dd = data;
-
-		while (g_io_channel_read_line(ioc, &msg, NULL, NULL, NULL) && msg != NULL)
-		{
-			/* translation hint:
-			 * Error while executing <spell command, e.g. "aspell"> (<error message>) */
-			dict_gui_status_add(dd, _("Error while executing \"%s\" (%s)."),
-				dd->spell_bin, g_strstrip(msg));
-			g_free(msg);
-		}
-	}
-	if (cond & (G_IO_ERR | G_IO_HUP | G_IO_NVAL))
-		return FALSE;
-
-	return TRUE;
-}
-
-
-static gboolean iofunc_write(GIOChannel *ioc, GIOCondition cond, gpointer data)
-{
-	if (NZV((gchar *) data))
-		g_io_channel_write_chars(ioc, (gchar *) data, -1, NULL, NULL);
-
-	g_free(data);
-
-	return FALSE;
-}
-
-
-void dict_aspell_start_query(DictData *dd, const gchar *word)
-{
-	GError  *error = NULL;
-	gchar  **argv;
-	gchar	*locale_cmd;
-	gint     stdout_fd;
-	gint     stderr_fd;
-	gint     stdin_fd;
-	guint	 i;
-	gsize	 tts_len;
-	gchar  **tts; /* text to search */
-	iodata	*iod;
-
-	if (! NZV(dd->spell_bin))
-	{
-		dict_gui_status_add(dd, _("Please set the aspell command in the preferences dialog."));
-		return;
-	}
-
-	if (! NZV(word))
-	{
-		dict_gui_status_add(dd, _("Invalid input."));
-		return;
-	}
-
-	tts = g_strsplit_set(word, " -_,.", 0);
-	tts_len = g_strv_length(tts);
-
-	for (i = 0; i < tts_len; i++)
-	{
-		locale_cmd = g_locale_from_utf8(dd->spell_bin, -1, NULL, NULL, NULL);
-		if (locale_cmd == NULL)
-			locale_cmd = g_strdup(dd->spell_bin);
-
-		argv = g_new0(gchar*, 5);
-		argv[0] = locale_cmd;
-		argv[1] = g_strdup("-a");
-		argv[2] = g_strdup("-l");
-		argv[3] = g_strdup(dd->spell_dictionary);
-		argv[4] = NULL;
-
-		if (g_spawn_async_with_pipes(NULL, argv, NULL,
-				G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD, NULL, NULL, NULL,
-				&stdin_fd, &stdout_fd, &stderr_fd, &error))
-		{
-			iod = g_new(iodata, 1);
-			iod->dd = dd;
-			iod->word = g_strdup(tts[i]);
-
-			set_up_io_channel(stdin_fd, G_IO_OUT, iofunc_write, g_strdup(tts[i]));
-			set_up_io_channel(stdout_fd, G_IO_IN|G_IO_PRI|G_IO_HUP|G_IO_ERR|G_IO_NVAL, iofunc_read, iod);
-			set_up_io_channel(stderr_fd, G_IO_IN|G_IO_PRI|G_IO_HUP|G_IO_ERR|G_IO_NVAL, iofunc_read_err, dd);
-			dict_gui_status_add(dd, _("Ready."));
-		}
-		else
-		{
-			dict_gui_status_add(dd, _("Process failed (%s)"), error->message);
-			g_error_free(error);
-			error = NULL;
-		}
-
-		g_strfreev(argv);
-	}
-
-	g_strfreev(tts);
-}
diff --git a/lib/aspell.h b/lib/aspell.h
deleted file mode 100644
index c5ab85a30e84e75cba5975d5c85b9ca9bf05e5be..0000000000000000000000000000000000000000
--- a/lib/aspell.h
+++ /dev/null
@@ -1,27 +0,0 @@
-/*  $Id$
- *
- *  Copyright 2006-2008 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
- *
- *  This program is free software; you can redistribute it and/or modify
- *  it under the terms of the GNU General Public License as published by
- *  the Free Software Foundation; either version 2 of the License, or
- *  (at your option) any later version.
- *
- *  This program 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 Library General Public License for more details.
- *
- *  You should have received a copy of the GNU General Public License
- *  along with this program; if not, write to the Free Software
- *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-
-
-#ifndef ASPELL_H
-#define ASPELL_H 1
-
-void dict_aspell_start_query(DictData *dd, const gchar *word);
-
-
-#endif
diff --git a/lib/prefs.c b/lib/prefs.c
index 9dbbcfd542dda4037a828f2e773c48219fc781ef..cb9e7375d69823df7cb32038acc0a12d1e9804eb 100644
--- a/lib/prefs.c
+++ b/lib/prefs.c
@@ -214,12 +214,60 @@ const gchar *dict_prefs_get_web_url_label(DictData *dd)
 }
 
 
-void color_set_cb(GtkColorButton *widget, GdkColor *color)
+static void color_set_cb(GtkColorButton *widget, GdkColor *color)
 {
 	gtk_color_button_get_color(widget, color);
 }
 
 
+static void button_dict_refresh_cb(GtkWidget *button, DictData *dd)
+{
+	GtkWidget *combo = g_object_get_data(G_OBJECT(button), "spell_combo");
+
+	dict_spell_get_dictionaries(dd, combo);
+}
+
+
+static gboolean spell_entry_focus_cb(GtkEntry *entry, GdkEventFocus *ev, GtkWidget *icon)
+{
+	gchar *path = g_find_program_in_path(gtk_entry_get_text(entry));
+
+	if (path != NULL)
+	{
+		gtk_image_set_from_stock(GTK_IMAGE(icon), GTK_STOCK_YES, GTK_ICON_SIZE_BUTTON);
+		g_free(path);
+	}
+	else
+		gtk_image_set_from_stock(GTK_IMAGE(icon), GTK_STOCK_STOP, GTK_ICON_SIZE_BUTTON);
+
+	return FALSE;
+}
+
+
+static void spell_entry_activate_cb(GtkEntry *entry, DictData *dd)
+{
+	GtkWidget *combo = g_object_get_data(G_OBJECT(entry), "spell_combo");
+	GtkWidget *icon = g_object_get_data(G_OBJECT(entry), "icon");
+
+	spell_entry_focus_cb(entry, NULL, icon);
+	dict_spell_get_dictionaries(dd, combo);
+}
+
+
+static void spell_combo_changed_cb(GtkComboBox *widget, DictData *dd)
+{
+	GtkTreeIter iter;
+
+	if (gtk_combo_box_get_active_iter(widget, &iter))
+	{
+		gchar *text;
+		gtk_tree_model_get(gtk_combo_box_get_model(widget), &iter, 0, &text, -1);
+		g_free(dd->spell_dictionary);
+		dd->spell_dictionary = text;
+	}
+}
+
+
 GtkWidget *dict_prefs_dialog_show(GtkWidget *parent, DictData *dd)
 {
 	GtkWidget *dialog, *inner_vbox, *notebook, *notebook_vbox;
@@ -564,7 +612,9 @@ GtkWidget *dict_prefs_dialog_show(GtkWidget *parent, DictData *dd)
 	 */
 #define PAGE_SPELL /* only for navigation in Geany's symbol list ;-) */
 	{
-		GtkWidget *table, *spell_entry, *spell_combo;
+		GtkWidget *table, *label_help, *spell_entry, *spell_combo, *button_refresh, *image, *icon;
+		GtkListStore *store;
+		GtkCellRenderer *renderer;
 
 		notebook_vbox = gtk_vbox_new(FALSE, 5);
 		gtk_widget_show(notebook_vbox);
@@ -577,45 +627,88 @@ GtkWidget *dict_prefs_dialog_show(GtkWidget *parent, DictData *dd)
 		label1 = gtk_label_new_with_mnemonic(_("Spell Check Program:"));
 		gtk_widget_show(label1);
 
+		icon = gtk_image_new();
+		gtk_widget_show(icon);
+
 		spell_entry = gtk_entry_new();
 		gtk_entry_set_max_length(GTK_ENTRY(spell_entry), 256);
 		if (dd->spell_bin != NULL)
 		{
 			gtk_entry_set_text(GTK_ENTRY(spell_entry), dd->spell_bin);
 		}
+		g_signal_connect(spell_entry, "focus-out-event", G_CALLBACK(spell_entry_focus_cb), icon);
+		g_signal_connect(spell_entry, "activate", G_CALLBACK(spell_entry_activate_cb), dd);
 		gtk_widget_show(spell_entry);
 
+		label_help = wrap_label_new(_(
+	"<i>The spell check program can be 'enchant', 'aspell', 'ispell' or any other spell check "
+	"program which is compatible to the ispell command.\nThe icon shows whether the entered "
+	"command exists.</i>"));
+		gtk_label_set_use_markup(GTK_LABEL(label_help), TRUE);
+		gtk_widget_show(label_help);
+
 		label2 = gtk_label_new_with_mnemonic(_("Dictionary:"));
 		gtk_widget_show(label2);
 
-		spell_combo = gtk_combo_box_new_text();
+		store = gtk_list_store_new(1, G_TYPE_STRING);
+		spell_combo = gtk_combo_box_new_with_model(GTK_TREE_MODEL(store));
+		g_object_set_data(G_OBJECT(spell_combo), "spell_entry", spell_entry);
+
+		renderer = gtk_cell_renderer_text_new();
+		gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(spell_combo), renderer, TRUE);
+		gtk_cell_layout_add_attribute(GTK_CELL_LAYOUT(spell_combo), renderer, "text", 0);
+
 		dict_spell_get_dictionaries(dd, spell_combo);
+		g_signal_connect(spell_combo, "changed", G_CALLBACK(spell_combo_changed_cb), dd);
 		gtk_widget_show(spell_combo);
 
+		button_refresh = gtk_button_new();
+		image = gtk_image_new_from_stock("gtk-refresh", GTK_ICON_SIZE_BUTTON);
+		gtk_button_set_image(GTK_BUTTON(button_refresh), image);
+		gtk_widget_show(button_refresh);
+		g_object_set_data(G_OBJECT(button_refresh), "spell_combo", spell_combo);
+		g_signal_connect(button_refresh, "clicked", G_CALLBACK(button_dict_refresh_cb), dd);
+
+		g_object_set_data(G_OBJECT(spell_entry), "icon", icon);
+		g_object_set_data(G_OBJECT(spell_entry), "spell_combo", spell_combo);
 		g_object_set_data(G_OBJECT(dialog), "spell_combo", spell_combo);
 		g_object_set_data(G_OBJECT(dialog), "spell_entry", spell_entry);
+		g_object_unref(store);
 
+		spell_entry_focus_cb(GTK_ENTRY(spell_entry), NULL, icon); /* initially set the icon */
 
-		table = gtk_table_new(2, 2, FALSE);
+		table = gtk_table_new(3, 3, FALSE);
 		gtk_widget_show(table);
 		gtk_table_set_row_spacings(GTK_TABLE(table), 5);
 		gtk_table_set_col_spacings(GTK_TABLE(table), 5);
 
-		gtk_table_attach(GTK_TABLE(table), label1, 0, 1, 0, 1,
+		gtk_table_attach(GTK_TABLE(table), label_help, 0, 3, 0, 1,
+						(GtkAttachOptions) (GTK_FILL | GTK_EXPAND),
+						(GtkAttachOptions) (0), 5, 5);
+
+		gtk_table_attach(GTK_TABLE(table), label1, 0, 1, 1, 2,
 						(GtkAttachOptions) (GTK_FILL),
 						(GtkAttachOptions) (0), 5, 5);
 		gtk_misc_set_alignment(GTK_MISC(label1), 1, 0);
 
-		gtk_table_attach(GTK_TABLE(table), spell_entry, 1, 2, 0, 1,
+		gtk_table_attach(GTK_TABLE(table), spell_entry, 1, 2, 1, 2,
 						(GtkAttachOptions) (GTK_FILL | GTK_EXPAND),
 						(GtkAttachOptions) (0), 5, 5);
 
-		gtk_table_attach(GTK_TABLE(table), label2, 0, 1, 1, 2,
+		gtk_table_attach(GTK_TABLE(table), icon, 2, 3, 1, 2,
+						(GtkAttachOptions) (GTK_FILL | GTK_EXPAND),
+						(GtkAttachOptions) (0), 5, 5);
+
+		gtk_table_attach(GTK_TABLE(table), label2, 0, 1, 2, 3,
 						(GtkAttachOptions) (GTK_FILL),
 						(GtkAttachOptions) (0), 5, 0);
 		gtk_misc_set_alignment(GTK_MISC(label2), 1, 0);
 
-		gtk_table_attach(GTK_TABLE(table), spell_combo, 1, 2, 1, 2,
+		gtk_table_attach(GTK_TABLE(table), spell_combo, 1, 2, 2, 3,
+						(GtkAttachOptions) (GTK_FILL | GTK_EXPAND),
+						(GtkAttachOptions) (0), 5, 5);
+
+		gtk_table_attach(GTK_TABLE(table), button_refresh, 2, 3, 2, 3,
 						(GtkAttachOptions) (GTK_FILL | GTK_EXPAND),
 						(GtkAttachOptions) (0), 5, 5);
 
diff --git a/lib/spell.c b/lib/spell.c
index b68ba81285e6638b2e6982fb940545a5ac9b31ce..0ff7e2dcb530755454373c170241d2ea88ca806b 100644
--- a/lib/spell.c
+++ b/lib/spell.c
@@ -300,18 +300,27 @@ static gchar **get_aspell_dicts(const gchar *str)
 
 void dict_spell_get_dictionaries(DictData *dd, GtkWidget *spell_combo)
 {
-	if (NZV(dd->spell_bin))
+	const gchar *entry_cmd = gtk_entry_get_text(
+		GTK_ENTRY(g_object_get_data(G_OBJECT(spell_combo), "spell_entry")));
+
+	if (*entry_cmd != '\0')
 	{
-		gchar *tmp = NULL, *cmd, *locale_cmd;
+		gchar *tmp = NULL;
+		gchar *cmd, *locale_cmd;
 		gboolean use_enchant = FALSE;
+		GtkTreeModel *model;
+		GtkTreeIter iter;
+
+		model = gtk_combo_box_get_model(GTK_COMBO_BOX(spell_combo));
+		gtk_list_store_clear(GTK_LIST_STORE(model));
 
-		if (strstr(dd->spell_bin, "enchant") != NULL)
+		if (strstr(entry_cmd, "enchant") != NULL)
 		{
 			cmd = g_strdup("enchant-lsmod -list-dicts");
 			use_enchant = TRUE;
 		}
 		else
-			cmd = g_strconcat(dd->spell_bin, " dump dicts", NULL);
+			cmd = g_strconcat(entry_cmd, " dump dicts", NULL);
 
 		locale_cmd = g_locale_from_utf8(cmd, -1, NULL, NULL, NULL);
 		if (locale_cmd == NULL)
@@ -330,9 +339,10 @@ void dict_spell_get_dictionaries(DictData *dd, GtkWidget *spell_combo)
 			{
 				if (NZV(list[i]))
 				{
-					gtk_combo_box_append_text(GTK_COMBO_BOX(spell_combo), list[i]);
+					gtk_list_store_append(GTK_LIST_STORE(model), &iter);
+					gtk_list_store_set(GTK_LIST_STORE(model), &iter, 0, list[i], -1);
 					if (strcmp(dd->spell_dictionary, list[i]) == 0)
-						gtk_combo_box_set_active(GTK_COMBO_BOX(spell_combo), i);
+						gtk_combo_box_set_active_iter(GTK_COMBO_BOX(spell_combo), &iter);
 				}
 			}
 			g_strfreev(list);