2 * (SLIK) SimpLIstic sKin functions
7 * This software is released under the GNU General Public License (GNU GPL).
8 * Please read the included file COPYING for more information.
9 * This software comes with no warranty of any kind, use at your own risk!
21 #include <sys/types.h>
26 #include <gdk-pixbuf/gdk-pixbuf.h>
28 #include "ui_tabcomp.h"
30 #include "ui_bookmark.h"
31 #include "ui_fileops.h"
32 #include "ui_spinner.h"
33 #include "ui_utildlg.h"
35 #include <gdk/gdkkeysyms.h> /* for key values */
38 /* define this to enable a pop-up menu that shows possible matches
39 * #define TAB_COMPLETION_ENABLE_POPUP_MENU
41 #define TAB_COMPLETION_ENABLE_POPUP_MENU 1
42 #define TAB_COMP_POPUP_MAX 500
44 #ifdef TAB_COMPLETION_ENABLE_POPUP_MENU
49 /* ----------------------------------------------------------------
50 Tab completion routines, can be connected to any gtkentry widget
51 using the tab_completion_add_to_entry() function.
52 Use remove_trailing_slash() to strip the trailing '/'.
53 ----------------------------------------------------------------*/
55 typedef struct _TabCompData TabCompData;
61 void (*enter_func)(const gchar *, gpointer);
62 void (*tab_func)(const gchar *, gpointer);
78 static void tab_completion_select_show(TabCompData *td);
80 static void tab_completion_free_list(TabCompData *td)
95 g_list_free(td->file_list);
99 static void tab_completion_read_dir(TabCompData *td, const gchar *path)
106 tab_completion_free_list(td);
108 pathl = path_from_utf8(path);
116 while ((dir = readdir(dp)) != NULL)
118 /* skips removed files */
121 gchar *name = dir->d_name;
122 if (strcmp(name, ".") != 0 && strcmp(name, "..") != 0)
124 list = g_list_prepend(list, path_to_utf8(name));
130 td->dir_path = g_strdup(path);
131 td->file_list = list;
134 static void tab_completion_destroy(GtkWidget *widget, gpointer data)
136 TabCompData *td = data;
138 tab_completion_free_list(td);
139 g_free(td->history_key);
141 if (td->fd) file_dialog_close(td->fd);
142 g_free(td->fd_title);
147 static gint tab_completion_emit_enter_signal(TabCompData *td)
150 if (!td->enter_func) return FALSE;
152 text = g_strdup(gtk_entry_get_text(GTK_ENTRY(td->entry)));
157 text = g_strconcat(homedir(), t + 1, NULL);
161 td->enter_func(text, td->enter_data);
167 static void tab_completion_emit_tab_signal(TabCompData *td)
170 if (!td->tab_func) return;
172 text = g_strdup(gtk_entry_get_text(GTK_ENTRY(td->entry)));
177 text = g_strconcat(homedir(), t + 1, NULL);
181 td->tab_func(text, td->tab_data);
185 #ifdef TAB_COMPLETION_ENABLE_POPUP_MENU
187 static gint tab_completion_popup_key_press(GtkWidget *widget, GdkEventKey *event, gpointer data)
189 TabCompData *td = data;
191 if (event->keyval == GDK_Tab ||
192 event->keyval == GDK_BackSpace ||
193 (event->keyval >= 0x20 && event->keyval <= 0xFF) )
195 if (event->keyval >= 0x20 && event->keyval <= 0xFF)
200 buf[0] = event->keyval;
202 gtk_editable_insert_text(GTK_EDITABLE(td->entry), buf, 1, &p);
203 gtk_editable_set_position(GTK_EDITABLE(td->entry), -1);
207 gtk_menu_popdown(GTK_MENU(widget));
208 /* doing this does not emit the "selection done" signal, unref it ourselves */
209 gtk_widget_unref(widget);
217 static void tab_completion_popup_cb(GtkWidget *widget, gpointer data)
224 td = g_object_get_data(G_OBJECT(widget), "tab_completion_data");
227 ptr = td->dir_path + strlen(td->dir_path) - 1;
228 buf = g_strconcat(td->dir_path, (ptr[0] == '/') ? "" : "/", name, NULL);
229 gtk_entry_set_text(GTK_ENTRY(td->entry), buf);
230 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(buf));
233 tab_completion_emit_tab_signal(td);
236 static void tab_completion_popup_pos_cb(GtkMenu *menu, gint *x, gint *y, gboolean *push_in, gpointer data)
238 TabCompData *td = data;
241 PangoRectangle strong_pos, weak_pos;
243 gint xoffset, yoffset;
246 gdk_window_get_origin(td->entry->window, x, y);
248 height = MIN(td->entry->requisition.height, td->entry->allocation.height);
251 length = strlen(gtk_entry_get_text(GTK_ENTRY(td->entry)));
252 gtk_entry_get_layout_offsets(GTK_ENTRY(td->entry), &xoffset, &yoffset);
254 layout = gtk_entry_get_layout(GTK_ENTRY(td->entry));
255 pango_layout_get_cursor_pos(layout, length, &strong_pos, &weak_pos);
256 *x += strong_pos.x / PANGO_SCALE + xoffset;
259 static void tab_completion_popup_list(TabCompData *td, GList *list)
272 * well, the menu would be too long anyway...
273 * (listing /dev causes gtk+ window allocation errors, -> too big a window)
274 * this is why menu popups are disabled, this really should be a popup scrollable listview.
276 if (g_list_length(list) > 200) return;
279 menu = popup_menu_short_lived();
282 while (work && count < TAB_COMP_POPUP_MAX)
284 gchar *name = work->data;
287 item = menu_item_add_simple(menu, name, G_CALLBACK(tab_completion_popup_cb), name);
288 g_object_set_data(G_OBJECT(item), "tab_completion_data", td);
294 g_signal_connect(G_OBJECT(menu), "key_press_event",
295 G_CALLBACK(tab_completion_popup_key_press), td);
297 /* peek at the current event to get the time, etc. */
298 event = gtk_get_current_event();
300 if (event && event->type == GDK_BUTTON_RELEASE)
302 ebutton = event->button.button;
311 etime = gdk_event_get_time(event);
312 gdk_event_free(event);
319 gtk_menu_popup(GTK_MENU(menu), NULL, NULL,
320 tab_completion_popup_pos_cb, td, ebutton, etime);
324 #define CASE_SORT strcmp
327 static gint simple_sort(gconstpointer a, gconstpointer b)
329 return CASE_SORT((gchar *)a, (gchar *)b);
334 static gint tab_completion_do(TabCompData *td)
336 const gchar *entry_text = gtk_entry_get_text(GTK_ENTRY(td->entry));
337 const gchar *entry_file;
340 gint home_exp = FALSE;
342 /* home dir expansion */
343 if (entry_text[0] == '~')
345 entry_dir = g_strconcat(homedir(), entry_text + 1, NULL);
350 entry_dir = g_strdup(entry_text);
353 entry_file = filename_from_path(entry_text);
355 if (isfile(entry_dir))
359 gtk_entry_set_text(GTK_ENTRY(td->entry), entry_dir);
360 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(entry_dir));
365 if (isdir(entry_dir) && strcmp(entry_file, ".") != 0 && strcmp(entry_file, "..") != 0)
367 ptr = entry_dir + strlen(entry_dir) - 1;
372 gtk_entry_set_text(GTK_ENTRY(td->entry), entry_dir);
373 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(entry_dir));
376 tab_completion_read_dir(td, entry_dir);
377 td->file_list = g_list_sort(td->file_list, simple_sort);
378 if (td->file_list && !td->file_list->next)
383 file = td->file_list->data;
384 buf = g_strconcat(entry_dir, file, NULL);
388 buf = g_strconcat(entry_dir, file, "/", NULL);
390 gtk_entry_set_text(GTK_ENTRY(td->entry), buf);
391 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(buf));
395 #ifdef TAB_COMPLETION_ENABLE_POPUP_MENU
399 tab_completion_popup_list(td, td->file_list);
408 gchar *buf = g_strconcat(entry_dir, "/", NULL);
409 gtk_entry_set_text(GTK_ENTRY(td->entry), buf);
410 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(buf));
417 ptr = (gchar *)filename_from_path(entry_dir);
418 if (ptr > entry_dir) ptr--;
421 if (strlen(entry_dir) == 0)
424 entry_dir = g_strdup("/");
427 if (isdir(entry_dir))
431 gint l = strlen(entry_file);
433 if (!td->dir_path || !td->file_list || strcmp(td->dir_path, entry_dir) != 0)
435 tab_completion_read_dir(td, entry_dir);
438 if (strcmp(entry_dir, "/") == 0) entry_dir[0] = '\0';
440 list = td->file_list;
443 gchar *file = list->data;
444 if (strncmp(entry_file, file, l) == 0)
446 poss = g_list_prepend(poss, file);
455 gchar *file = poss->data;
458 buf = g_strconcat(entry_dir, "/", file, NULL);
463 buf = g_strconcat(entry_dir, "/", file, "/", NULL);
465 gtk_entry_set_text(GTK_ENTRY(td->entry), buf);
466 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(buf));
474 gint c = strlen(entry_file);
476 gchar *test_file = poss->data;
481 if (!list) done = TRUE;
484 gchar *file = list->data;
485 if (strlen(file) < c || strncmp(test_file, file, c) != 0)
498 file = g_strdup(test_file);
500 buf = g_strconcat(entry_dir, "/", file, NULL);
501 gtk_entry_set_text(GTK_ENTRY(td->entry), buf);
502 gtk_editable_set_position(GTK_EDITABLE(td->entry), strlen(buf));
504 #ifdef TAB_COMPLETION_ENABLE_POPUP_MENU
506 poss = g_list_sort(poss, simple_sort);
507 tab_completion_popup_list(td, poss);
527 static gint tab_completion_key_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data)
529 TabCompData *td = data;
530 gint stop_signal = FALSE;
532 switch (event->keyval)
535 if (!(event->state & GDK_CONTROL_MASK))
537 if (tab_completion_do(td))
539 tab_completion_emit_tab_signal(td);
544 case GDK_Return: case GDK_KP_Enter:
546 (event->state & GDK_CONTROL_MASK))
548 tab_completion_select_show(td);
551 else if (tab_completion_emit_enter_signal(td))
560 if (stop_signal) g_signal_stop_emission_by_name(G_OBJECT(widget), "key_press_event");
562 return (stop_signal);
565 static void tab_completion_button_pressed(GtkWidget *widget, gpointer data)
568 GtkWidget *entry = data;
570 td = g_object_get_data(G_OBJECT(entry), "tab_completion_data");
574 if (!GTK_WIDGET_HAS_FOCUS(entry))
576 gtk_widget_grab_focus(entry);
579 if (tab_completion_do(td))
581 tab_completion_emit_tab_signal(td);
585 static void tab_completion_button_size_allocate(GtkWidget *button, GtkAllocation *allocation, gpointer data)
587 GtkWidget *parent = data;
589 if (allocation->height > parent->allocation.height)
591 GtkAllocation button_allocation;
593 button_allocation = button->allocation;
594 button_allocation.height = parent->allocation.height;
595 button_allocation.y = parent->allocation.y +
596 (parent->allocation.height - parent->allocation.height) / 2;
597 gtk_widget_size_allocate(button, &button_allocation);
601 static GtkWidget *tab_completion_create_complete_button(GtkWidget *entry, GtkWidget *parent)
607 button = gtk_button_new();
608 GTK_WIDGET_UNSET_FLAGS(button, GTK_CAN_FOCUS);
609 g_signal_connect(G_OBJECT(button), "size_allocate",
610 G_CALLBACK(tab_completion_button_size_allocate), parent);
611 g_signal_connect(G_OBJECT(button), "clicked",
612 G_CALLBACK(tab_completion_button_pressed), entry);
614 pixbuf = gdk_pixbuf_new_from_inline(-1, icon_tabcomp, FALSE, NULL);
615 icon = gtk_image_new_from_pixbuf(pixbuf);
616 gdk_pixbuf_unref(pixbuf);
617 gtk_container_add(GTK_CONTAINER(button), icon);
618 gtk_widget_show(icon);
624 *----------------------------------------------------------------------------
626 *----------------------------------------------------------------------------
629 GtkWidget *tab_completion_new_with_history(GtkWidget **entry, const gchar *text,
630 const gchar *history_key, gint max_levels,
631 void (*enter_func)(const gchar *, gpointer), gpointer data)
635 GtkWidget *combo_entry;
641 box = gtk_hbox_new(FALSE, 0);
643 combo = gtk_combo_box_entry_new_text();
644 gtk_box_pack_start(GTK_BOX(box), combo, TRUE, TRUE, 0);
645 gtk_widget_show(combo);
647 combo_entry = GTK_BIN(combo)->child;
649 gtk_combo_set_case_sensitive(GTK_COMBO(combo), TRUE);
650 gtk_combo_set_use_arrows(GTK_COMBO(combo), FALSE);
653 button = tab_completion_create_complete_button(combo_entry, combo);
654 gtk_box_pack_start(GTK_BOX(box), button, FALSE, FALSE, 0);
655 gtk_widget_show(button);
657 tab_completion_add_to_entry(combo_entry, enter_func, data);
659 td = g_object_get_data(G_OBJECT(combo_entry), "tab_completion_data");
660 if (!td) return NULL; /* this should never happen! */
663 td->has_history = TRUE;
664 td->history_key = g_strdup(history_key);
665 td->history_levels = max_levels;
667 work = history_list_get_by_key(td->history_key);
669 work = history_list_get_by_key(history_key);
672 gtk_combo_box_append_text(GTK_COMBO_BOX(combo), (gchar *)work->data);
679 gtk_entry_set_text(GTK_ENTRY(combo_entry), text);
683 gtk_combo_box_set_active(GTK_COMBO_BOX(combo), 0);
686 if (entry) *entry = combo_entry;
690 const gchar *tab_completion_set_to_last_history(GtkWidget *entry)
692 TabCompData *td = g_object_get_data(G_OBJECT(entry), "tab_completion_data");
695 if (!td || !td->has_history) return NULL;
697 buf = history_list_find_last_path_by_key(td->history_key);
700 gtk_entry_set_text(GTK_ENTRY(td->entry), buf);
706 void tab_completion_append_to_history(GtkWidget *entry, const gchar *path)
712 td = g_object_get_data(G_OBJECT(entry), "tab_completion_data");
716 if (!td || !td->has_history) return;
718 history_list_add_to_key(td->history_key, path, td->history_levels);
720 gtk_combo_box_set_active(GTK_COMBO_BOX(td->combo), -1);
722 store = gtk_combo_box_get_model(GTK_COMBO_BOX(td->combo));
723 gtk_list_store_clear(GTK_LIST_STORE(store));
725 work = history_list_get_by_key(td->history_key);
728 gtk_combo_box_append_text(GTK_COMBO_BOX(td->combo), (gchar *)work->data);
733 GtkWidget *tab_completion_new(GtkWidget **entry, const gchar *text,
734 void (*enter_func)(const gchar *, gpointer), gpointer data)
740 hbox = gtk_hbox_new(FALSE, 0);
742 newentry = gtk_entry_new();
743 if (text) gtk_entry_set_text(GTK_ENTRY(newentry), text);
744 gtk_box_pack_start(GTK_BOX(hbox), newentry, TRUE, TRUE, 0);
745 gtk_widget_show(newentry);
747 button = tab_completion_create_complete_button(newentry, newentry);
748 gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 0);
749 gtk_widget_show(button);
751 tab_completion_add_to_entry(newentry, enter_func, data);
753 if (entry) *entry = newentry;
757 void tab_completion_add_to_entry(GtkWidget *entry, void (*enter_func)(const gchar *, gpointer), gpointer data)
762 printf("Tab completion error: entry != NULL\n");
766 td = g_new0(TabCompData, 1);
769 td->file_list = NULL;
770 td->enter_func = enter_func;
771 td->enter_data = data;
775 td->has_history = FALSE;
776 td->history_key = NULL;
777 td->history_levels = 0;
779 g_object_set_data(G_OBJECT(td->entry), "tab_completion_data", td);
781 g_signal_connect(G_OBJECT(entry), "key_press_event",
782 G_CALLBACK(tab_completion_key_pressed), td);
783 g_signal_connect(G_OBJECT(entry), "destroy",
784 G_CALLBACK(tab_completion_destroy), td);
787 void tab_completion_add_tab_func(GtkWidget *entry, void (*tab_func)(const gchar *, gpointer), gpointer data)
789 TabCompData *td = g_object_get_data(G_OBJECT(entry), "tab_completion_data");
793 td->tab_func = tab_func;
797 gchar *remove_trailing_slash(const gchar *path)
801 if (!path) return NULL;
803 ret = g_strdup(path);
805 if (l > 1 && ret[l - 1] == '/') ret[l - 1] = '\0';
810 static void tab_completion_select_cancel_cb(FileDialog *fd, gpointer data)
812 TabCompData *td = data;
815 file_dialog_close(fd);
818 static void tab_completion_select_ok_cb(FileDialog *fd, gpointer data)
820 TabCompData *td = data;
822 gtk_entry_set_text(GTK_ENTRY(td->entry), gtk_entry_get_text(GTK_ENTRY(fd->entry)));
824 tab_completion_select_cancel_cb(fd, data);
826 tab_completion_emit_enter_signal(td);
829 static void tab_completion_select_show(TabCompData *td)
836 gtk_window_present(GTK_WINDOW(GENERIC_DIALOG(td->fd)->dialog));
840 title = (td->fd_title) ? td->fd_title : _("Select path");
841 td->fd = file_dialog_new(title, PACKAGE, "select_path", td->entry,
842 tab_completion_select_cancel_cb, td);
843 file_dialog_add_button(td->fd, GTK_STOCK_OK, NULL,
844 tab_completion_select_ok_cb, TRUE);
846 generic_dialog_add_message(GENERIC_DIALOG(td->fd), NULL, title, NULL);
848 path = gtk_entry_get_text(GTK_ENTRY(td->entry));
849 if (strlen(path) == 0) path = NULL;
850 if (td->fd_folders_only)
852 file_dialog_add_path_widgets(td->fd, NULL, path, td->history_key, NULL, NULL);
856 file_dialog_add_path_widgets(td->fd, NULL, path, td->history_key, "*", _("All files"));
859 gtk_widget_show(GENERIC_DIALOG(td->fd)->dialog);
862 static void tab_completion_select_pressed(GtkWidget *widget, gpointer data)
864 TabCompData *td = data;
866 tab_completion_select_show(td);
869 void tab_completion_add_select_button(GtkWidget *entry, const gchar *title, gint folders_only)
875 td = g_object_get_data(G_OBJECT(entry), "tab_completion_data");
879 g_free(td->fd_title);
880 td->fd_title = g_strdup(title);
881 td->fd_folders_only = folders_only;
883 if (td->fd_button) return;
885 parent = (td->combo) ? td->combo : td->entry;
887 hbox = gtk_widget_get_parent(parent);
888 if (!GTK_IS_BOX(hbox)) return;
890 td->fd_button = gtk_button_new_with_label("...");
891 g_signal_connect(G_OBJECT(td->fd_button), "size_allocate",
892 G_CALLBACK(tab_completion_button_size_allocate), parent);
893 g_signal_connect(G_OBJECT(td->fd_button), "clicked",
894 G_CALLBACK(tab_completion_select_pressed), td);
896 gtk_box_pack_start(GTK_BOX(hbox), td->fd_button, FALSE, FALSE, 0);
898 gtk_widget_show(td->fd_button);