4 * Copyright (C) 2008 - 2009 The Geeqie Team
8 * This software is released under the GNU General Public License (GNU GPL).
9 * Please read the included file COPYING for more information.
10 * This software comes with no warranty of any kind, use at your own risk!
13 #include <glib/gprintf.h>
16 #include "bar_keywords.h"
19 #include "history_list.h"
22 #include "ui_fileops.h"
24 #include "ui_utildlg.h"
33 static void bar_pane_keywords_keyword_update_all(void);
34 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data);
37 *-------------------------------------------------------------------
38 * keyword / comment utils
39 *-------------------------------------------------------------------
43 GList *keyword_list_pull(GtkWidget *text_widget)
48 text = text_widget_text_pull(text_widget);
49 list = string_to_keywords_list(text);
56 /* the "changed" signal should be blocked before calling this */
57 static void keyword_list_push(GtkWidget *textview, GList *list)
59 GtkTextBuffer *buffer;
60 GtkTextIter start, end;
62 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(textview));
63 gtk_text_buffer_get_bounds(buffer, &start, &end);
64 gtk_text_buffer_delete(buffer, &start, &end);
68 const gchar *word = list->data;
71 gtk_text_buffer_get_end_iter(buffer, &iter);
72 if (word) gtk_text_buffer_insert(buffer, &iter, word, -1);
73 gtk_text_buffer_get_end_iter(buffer, &iter);
74 gtk_text_buffer_insert(buffer, &iter, "\n", -1);
82 *-------------------------------------------------------------------
84 *-------------------------------------------------------------------
89 FILTER_KEYWORD_COLUMN_TOGGLE = 0,
90 FILTER_KEYWORD_COLUMN_MARK,
91 FILTER_KEYWORD_COLUMN_NAME,
92 FILTER_KEYWORD_COLUMN_IS_KEYWORD,
93 FILTER_KEYWORD_COLUMN_COUNT
96 static GType filter_keyword_column_types[] = {G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN};
98 typedef struct _PaneKeywordsData PaneKeywordsData;
99 struct _PaneKeywordsData
104 GtkWidget *keyword_view;
105 GtkWidget *keyword_treeview;
107 GtkTreePath *click_tpath;
109 gboolean expand_checked;
110 gboolean collapse_unchecked;
111 gboolean hide_unchecked;
117 typedef struct _ConfDialogData ConfDialogData;
118 struct _ConfDialogData
120 PaneKeywordsData *pkd;
121 GtkTreePath *click_tpath;
125 GtkWidget *edit_widget;
128 gboolean edit_existing;
131 static GList *bar_list = NULL;
134 static void bar_pane_keywords_write(PaneKeywordsData *pkd)
138 if (!pkd->fd) return;
140 list = keyword_list_pull(pkd->keyword_view);
142 metadata_write_list(pkd->fd, KEYWORD_KEY, list);
144 string_list_free(list);
147 gboolean bar_keyword_tree_expand_if_set_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
149 PaneKeywordsData *pkd = data;
152 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set, -1);
154 if (set && !gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
156 gtk_tree_view_expand_to_path(GTK_TREE_VIEW(pkd->keyword_treeview), path);
161 gboolean bar_keyword_tree_collapse_if_unset_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
163 PaneKeywordsData *pkd = data;
167 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set,
168 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
170 if (is_keyword && !set && gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
172 gtk_tree_view_collapse_row(GTK_TREE_VIEW(pkd->keyword_treeview), path);
177 static void bar_keyword_tree_sync(PaneKeywordsData *pkd)
181 GtkTreeModel *keyword_tree;
184 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
185 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
187 keywords = keyword_list_pull(pkd->keyword_view);
188 keyword_show_set_in(GTK_TREE_STORE(keyword_tree), model, keywords);
189 if (pkd->hide_unchecked) keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
190 string_list_free(keywords);
192 gtk_tree_model_filter_refilter(GTK_TREE_MODEL_FILTER(model));
194 if (pkd->expand_checked) gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
195 if (pkd->collapse_unchecked) gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
198 static void bar_pane_keywords_keyword_update_all(void)
205 PaneKeywordsData *pkd;
211 bar_keyword_tree_sync(pkd);
215 static void bar_pane_keywords_update(PaneKeywordsData *pkd)
217 GList *keywords = NULL;
218 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
220 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
222 keywords = metadata_read_list(pkd->fd, KEYWORD_KEY, METADATA_PLAIN);
223 keyword_list_push(pkd->keyword_view, keywords);
224 bar_keyword_tree_sync(pkd);
225 string_list_free(keywords);
227 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
231 void bar_pane_keywords_set_fd(GtkWidget *pane, FileData *fd)
233 PaneKeywordsData *pkd;
235 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
238 file_data_unref(pkd->fd);
239 pkd->fd = file_data_ref(fd);
241 bar_pane_keywords_update(pkd);
244 static void bar_pane_keywords_write_config(GtkWidget *pane, GString *outstr, gint indent)
246 PaneKeywordsData *pkd;
248 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
251 WRITE_STRING("<pane_keywords\n");
253 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(pkd->pane.title)));
254 WRITE_BOOL(*pkd, pane.expanded);
255 WRITE_CHAR(*pkd, key);
257 WRITE_STRING("/>\n");
260 gint bar_pane_keywords_event(GtkWidget *bar, GdkEvent *event)
262 PaneKeywordsData *pkd;
264 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
265 if (!pkd) return FALSE;
267 if (GTK_WIDGET_HAS_FOCUS(pkd->keyword_view)) return gtk_widget_event(pkd->keyword_view, event);
272 static void bar_pane_keywords_keyword_toggle(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
274 PaneKeywordsData *pkd = data;
280 GtkTreeIter child_iter;
281 GtkTreeModel *keyword_tree;
283 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
285 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
287 tpath = gtk_tree_path_new_from_string(path);
288 gtk_tree_model_get_iter(model, &iter, tpath);
289 gtk_tree_path_free(tpath);
291 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_TOGGLE, &active, -1);
295 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
296 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
298 list = keyword_list_pull(pkd->keyword_view);
300 keyword_tree_set(keyword_tree, &child_iter, &list);
302 keyword_tree_reset(keyword_tree, &child_iter, &list);
304 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
305 keyword_list_push(pkd->keyword_view, list);
306 string_list_free(list);
307 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
309 /* call this just once in the end */
310 bar_pane_keywords_changed(keyword_buffer, pkd);
312 bar_pane_keywords_change calls bar_keyword_tree_sync, no need to do it again
313 bar_keyword_tree_sync(pkd);
317 void bar_pane_keywords_filter_modify(GtkTreeModel *model, GtkTreeIter *iter, GValue *value, gint column, gpointer data)
319 PaneKeywordsData *pkd = data;
320 GtkTreeModel *keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
321 GtkTreeIter child_iter;
323 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, iter);
325 memset(value, 0, sizeof (GValue));
329 case FILTER_KEYWORD_COLUMN_TOGGLE:
331 GList *keywords = keyword_list_pull(pkd->keyword_view);
332 gboolean set = keyword_tree_is_set(keyword_tree, &child_iter, keywords);
333 string_list_free(keywords);
335 g_value_init(value, G_TYPE_BOOLEAN);
336 g_value_set_boolean(value, set);
339 case FILTER_KEYWORD_COLUMN_MARK:
340 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_MARK, value);
342 case FILTER_KEYWORD_COLUMN_NAME:
343 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_NAME, value);
345 case FILTER_KEYWORD_COLUMN_IS_KEYWORD:
346 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_IS_KEYWORD, value);
353 gboolean bar_pane_keywords_filter_visible(GtkTreeModel *keyword_tree, GtkTreeIter *iter, gpointer data)
355 GtkTreeModel *filter = data;
357 return !keyword_is_hidden_in(keyword_tree, iter, filter);
360 static void bar_pane_keywords_set_selection(PaneKeywordsData *pkd, gboolean append)
362 GList *keywords = NULL;
366 keywords = keyword_list_pull(pkd->keyword_view);
368 list = layout_selection_list(pkd->pane.lw);
372 FileData *fd = work->data;
377 metadata_append_list(fd, KEYWORD_KEY, keywords);
381 metadata_write_list(fd, KEYWORD_KEY, keywords);
386 string_list_free(keywords);
389 static void bar_pane_keywords_sel_add_cb(GtkWidget *button, gpointer data)
391 PaneKeywordsData *pkd = data;
393 bar_pane_keywords_set_selection(pkd, TRUE);
396 static void bar_pane_keywords_sel_replace_cb(GtkWidget *button, gpointer data)
398 PaneKeywordsData *pkd = data;
400 bar_pane_keywords_set_selection(pkd, FALSE);
403 static void bar_pane_keywords_populate_popup_cb(GtkTextView *textview, GtkMenu *menu, gpointer data)
405 PaneKeywordsData *pkd = data;
407 menu_item_add_divider(GTK_WIDGET(menu));
408 menu_item_add_stock(GTK_WIDGET(menu), _("Add keywords to selected files"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_keywords_sel_add_cb), pkd);
409 menu_item_add_stock(GTK_WIDGET(menu), _("Replace existing keywords in selected files"), GTK_STOCK_CONVERT, G_CALLBACK(bar_pane_keywords_sel_replace_cb), pkd);
413 static void bar_pane_keywords_notify_cb(FileData *fd, NotifyType type, gpointer data)
415 PaneKeywordsData *pkd = data;
416 if ((type & (NOTIFY_REREAD | NOTIFY_CHANGE | NOTIFY_METADATA)) && fd == pkd->fd) bar_pane_keywords_update(pkd);
419 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data)
421 PaneKeywordsData *pkd = data;
423 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
424 bar_pane_keywords_write(pkd);
425 bar_keyword_tree_sync(pkd);
426 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
431 *-------------------------------------------------------------------
433 *-------------------------------------------------------------------
437 static GtkTargetEntry bar_pane_keywords_drag_types[] = {
438 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
439 { "text/plain", 0, TARGET_TEXT_PLAIN }
441 static gint n_keywords_drag_types = 2;
444 static GtkTargetEntry bar_pane_keywords_drop_types[] = {
445 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
446 { "text/plain", 0, TARGET_TEXT_PLAIN }
448 static gint n_keywords_drop_types = 2;
451 static void bar_pane_keywords_dnd_get(GtkWidget *tree_view, GdkDragContext *context,
452 GtkSelectionData *selection_data, guint info,
453 guint time, gpointer data)
457 GtkTreeIter child_iter;
458 GtkTreeModel *keyword_tree;
460 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
462 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
464 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
465 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
469 case TARGET_APP_KEYWORD_PATH:
471 GList *path = keyword_tree_get_path(keyword_tree, &child_iter);
472 gtk_selection_data_set(selection_data, selection_data->target,
473 8, (gpointer) &path, sizeof(path));
477 case TARGET_TEXT_PLAIN:
480 gchar *name = keyword_get_name(keyword_tree, &child_iter);
481 gtk_selection_data_set_text(selection_data, name, -1);
488 static void bar_pane_keywords_dnd_begin(GtkWidget *tree_view, GdkDragContext *context, gpointer data)
492 GtkTreeIter child_iter;
493 GtkTreeModel *keyword_tree;
496 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
498 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
500 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
501 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
503 name = keyword_get_name(keyword_tree, &child_iter);
505 dnd_set_drag_label(tree_view, context, name);
510 static void bar_pane_keywords_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
515 static gboolean bar_pane_keywords_dnd_can_move(GtkTreeModel *keyword_tree, GtkTreeIter *src_kw_iter, GtkTreeIter *dest_kw_iter)
520 if (dest_kw_iter && keyword_same_parent(keyword_tree, src_kw_iter, dest_kw_iter))
522 return TRUE; /* reordering of siblings is ok */
524 if (!dest_kw_iter && !gtk_tree_model_iter_parent(keyword_tree, &parent, src_kw_iter))
526 return TRUE; /* reordering of top-level siblings is ok */
529 src_name = keyword_get_name(keyword_tree, src_kw_iter);
530 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, src_name, FALSE))
539 static gboolean bar_pane_keywords_dnd_skip_existing(GtkTreeModel *keyword_tree, GtkTreeIter *dest_kw_iter, GList **keywords)
541 /* we have to find at least one keyword that does not already exist as a sibling of dest_kw_iter */
542 GList *work = *keywords;
545 gchar *keyword = work->data;
546 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, keyword, FALSE))
548 GList *next = work->next;
550 *keywords = g_list_delete_link(*keywords, work);
561 static void bar_pane_keywords_dnd_receive(GtkWidget *tree_view, GdkDragContext *context,
563 GtkSelectionData *selection_data, guint info,
564 guint time, gpointer data)
566 PaneKeywordsData *pkd = data;
567 GtkTreePath *tpath = NULL;
568 GtkTreeViewDropPosition pos;
571 GtkTreeModel *keyword_tree;
572 gboolean src_valid = FALSE;
573 GList *new_keywords = NULL;
576 /* iterators for keyword_tree */
577 GtkTreeIter src_kw_iter;
578 GtkTreeIter dest_kw_iter;
579 GtkTreeIter new_kw_iter;
581 g_signal_stop_emission_by_name(tree_view, "drag_data_received");
583 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
584 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
586 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
587 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), NULL, pos);
591 case TARGET_APP_KEYWORD_PATH:
593 GList *path = *(gpointer *)selection_data->data;
594 src_valid = keyword_tree_get_iter(keyword_tree, &src_kw_iter, path);
595 string_list_free(path);
599 new_keywords = string_to_keywords_list((gchar *)selection_data->data);
605 GtkTreeIter dest_iter;
606 gtk_tree_model_get_iter(model, &dest_iter, tpath);
607 gtk_tree_path_free(tpath);
608 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &dest_kw_iter, &dest_iter);
610 if (src_valid && gtk_tree_store_is_ancestor(GTK_TREE_STORE(keyword_tree), &src_kw_iter, &dest_kw_iter))
612 /* can't move to it's own child */
616 if (src_valid && keyword_compare(keyword_tree, &src_kw_iter, &dest_kw_iter) == 0)
618 /* can't move to itself */
622 if ((pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE || pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER) &&
623 !gtk_tree_model_iter_has_child(keyword_tree, &dest_kw_iter))
625 /* the node has no children, all keywords can be added */
626 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &dest_kw_iter);
630 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, &dest_kw_iter))
632 /* the keyword can't be moved if the same name already exist */
635 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, &dest_kw_iter, &new_keywords))
637 /* the keywords can't be added if the same name already exist */
643 case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
644 case GTK_TREE_VIEW_DROP_BEFORE:
645 gtk_tree_store_insert_before(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
647 case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
648 case GTK_TREE_VIEW_DROP_AFTER:
649 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
657 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, NULL))
659 /* the keyword can't be moved if the same name already exist */
662 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, NULL, &new_keywords))
664 /* the keywords can't be added if the same name already exist */
667 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL);
673 keyword_move_recursive(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &src_kw_iter);
679 gchar *keyword = work->data;
680 keyword_set(GTK_TREE_STORE(keyword_tree), &new_kw_iter, keyword, TRUE);
686 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &new_kw_iter);
690 string_list_free(new_keywords);
691 bar_keyword_tree_sync(pkd);
694 static gint bar_pane_keywords_dnd_motion(GtkWidget *tree_view, GdkDragContext *context,
695 gint x, gint y, guint time, gpointer data)
697 GtkTreePath *tpath = NULL;
698 GtkTreeViewDropPosition pos;
699 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
703 GtkTreeIter dest_iter;
704 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
705 gtk_tree_model_get_iter(model, &dest_iter, tpath);
706 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE && gtk_tree_model_iter_has_child(model, &dest_iter))
707 pos = GTK_TREE_VIEW_DROP_BEFORE;
709 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER && gtk_tree_model_iter_has_child(model, &dest_iter))
710 pos = GTK_TREE_VIEW_DROP_AFTER;
713 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), tpath, pos);
714 gtk_tree_path_free(tpath);
716 if (tree_view == gtk_drag_get_source_widget(context))
717 gdk_drag_status(context, GDK_ACTION_MOVE, time);
719 gdk_drag_status(context, GDK_ACTION_COPY, time);
725 *-------------------------------------------------------------------
727 *-------------------------------------------------------------------
730 static void bar_pane_keywords_edit_destroy_cb(GtkWidget *widget, gpointer data)
732 ConfDialogData *cdd = data;
733 gtk_tree_path_free(cdd->click_tpath);
738 static void bar_pane_keywords_edit_cancel_cb(GenericDialog *gd, gpointer data)
743 static void bar_pane_keywords_edit_ok_cb(GenericDialog *gd, gpointer data)
745 ConfDialogData *cdd = data;
746 PaneKeywordsData *pkd = cdd->pkd;
749 GtkTreeModel *keyword_tree;
752 gboolean have_dest = FALSE;
756 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
757 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
759 if (cdd->click_tpath)
762 if (gtk_tree_model_get_iter(model, &iter, cdd->click_tpath))
764 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
769 if (cdd->edit_existing && !have_dest) return;
771 keywords = keyword_list_pull(cdd->edit_widget);
773 if (cdd->edit_existing)
775 if (keywords && keywords->data && /* there should be one keyword */
776 !keyword_exists(keyword_tree, NULL, &kw_iter, keywords->data, TRUE))
778 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, keywords->data, cdd->is_keyword);
783 GList *work = keywords;
788 if (keyword_exists(keyword_tree, NULL, have_dest ? &kw_iter : NULL, work->data, FALSE))
795 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &kw_iter);
799 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &add, NULL);
803 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, work->data, cdd->is_keyword);
807 string_list_free(keywords);
810 static void bar_pane_keywords_conf_set_helper(GtkWidget *widget, gpointer data)
812 ConfDialogData *cdd = data;
813 cdd->is_keyword = FALSE;
816 static void bar_pane_keywords_conf_set_kw(GtkWidget *widget, gpointer data)
818 ConfDialogData *cdd = data;
819 cdd->is_keyword = TRUE;
824 static void bar_pane_keywords_edit_dialog(PaneKeywordsData *pkd, gboolean edit_existing)
833 gboolean is_keyword = TRUE;
836 if (edit_existing && pkd->click_tpath)
840 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
842 if (gtk_tree_model_get_iter(model, &iter, pkd->click_tpath))
844 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
845 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
853 if (edit_existing && !name) return;
856 cdd = g_new0(ConfDialogData, 1);
858 cdd->click_tpath = pkd->click_tpath;
859 pkd->click_tpath = NULL;
860 cdd->is_keyword = is_keyword;
861 cdd->edit_existing = edit_existing;
863 cdd->gd = gd = generic_dialog_new(name ? _("Edit keyword") : _("Add keywords"), "keyword_edit",
865 bar_pane_keywords_edit_cancel_cb, cdd);
866 g_signal_connect(G_OBJECT(gd->dialog), "destroy",
867 G_CALLBACK(bar_pane_keywords_edit_destroy_cb), cdd);
870 generic_dialog_add_message(gd, NULL, name ? _("Configure keyword") : _("Add keyword"), NULL);
872 generic_dialog_add_button(gd, GTK_STOCK_OK, NULL,
873 bar_pane_keywords_edit_ok_cb, TRUE);
875 table = pref_table_new(gd->vbox, 3, 1, FALSE, TRUE);
876 pref_table_label(table, 0, 0, _("Keyword:"), 1.0);
877 cdd->edit_widget = gtk_entry_new();
878 gtk_widget_set_size_request(cdd->edit_widget, 300, -1);
879 if (name) gtk_entry_set_text(GTK_ENTRY(cdd->edit_widget), name);
880 gtk_table_attach_defaults(GTK_TABLE(table), cdd->edit_widget, 1, 2, 0, 1);
881 /* here could eventually be a text view instead of entry */
882 generic_dialog_attach_default(gd, cdd->edit_widget);
883 gtk_widget_show(cdd->edit_widget);
885 group = pref_group_new(gd->vbox, FALSE, _("Keyword type:"), GTK_ORIENTATION_VERTICAL);
887 button = pref_radiobutton_new(group, NULL, _("Active keyword"),
889 G_CALLBACK(bar_pane_keywords_conf_set_kw), cdd);
890 button = pref_radiobutton_new(group, button, _("Helper"),
892 G_CALLBACK(bar_pane_keywords_conf_set_helper), cdd);
895 gtk_widget_show(gd->dialog);
902 *-------------------------------------------------------------------
904 *-------------------------------------------------------------------
907 static void bar_pane_keywords_edit_dialog_cb(GtkWidget *menu_widget, gpointer data)
909 PaneKeywordsData *pkd = data;
910 bar_pane_keywords_edit_dialog(pkd, TRUE);
913 static void bar_pane_keywords_add_dialog_cb(GtkWidget *menu_widget, gpointer data)
915 PaneKeywordsData *pkd = data;
916 bar_pane_keywords_edit_dialog(pkd, FALSE);
919 static void bar_pane_keywords_connect_mark_cb(GtkWidget *menu_widget, gpointer data)
921 PaneKeywordsData *pkd = data;
926 GtkTreeModel *keyword_tree;
929 gint mark = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menu_widget), "mark")) - 1;
931 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
932 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
934 if (!pkd->click_tpath) return;
935 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
937 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
939 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
941 meta_data_connect_mark_with_keyword(keyword_tree, &kw_iter, mark);
943 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
944 // bar_pane_keywords_update(pkd);
948 static void bar_pane_keywords_delete_cb(GtkWidget *menu_widget, gpointer data)
950 PaneKeywordsData *pkd = data;
954 GtkTreeModel *keyword_tree;
957 if (!pkd->click_tpath) return;
959 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
960 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
962 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
963 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
965 keyword_delete(GTK_TREE_STORE(keyword_tree), &kw_iter);
968 static void bar_pane_keywords_hide_cb(GtkWidget *menu_widget, gpointer data)
970 PaneKeywordsData *pkd = data;
974 GtkTreeModel *keyword_tree;
977 if (!pkd->click_tpath) return;
979 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
980 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
982 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
983 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
985 keyword_hide_in(GTK_TREE_STORE(keyword_tree), &kw_iter, model);
988 static void bar_pane_keywords_show_all_cb(GtkWidget *menu_widget, gpointer data)
990 PaneKeywordsData *pkd = data;
993 GtkTreeModel *keyword_tree;
995 pkd->hide_unchecked = FALSE;
997 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
998 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1000 keyword_show_all_in(GTK_TREE_STORE(keyword_tree), model);
1002 if (!pkd->collapse_unchecked) gtk_tree_view_expand_all(GTK_TREE_VIEW(pkd->keyword_treeview));
1003 bar_keyword_tree_sync(pkd);
1006 static void bar_pane_keywords_expand_checked_cb(GtkWidget *menu_widget, gpointer data)
1008 PaneKeywordsData *pkd = data;
1009 GtkTreeModel *model;
1011 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1012 gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
1015 static void bar_pane_keywords_collapse_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1017 PaneKeywordsData *pkd = data;
1018 GtkTreeModel *model;
1020 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1021 gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
1024 static void bar_pane_keywords_hide_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1026 PaneKeywordsData *pkd = data;
1027 GtkTreeModel *model;
1029 GtkTreeModel *keyword_tree;
1032 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1033 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1035 keywords = keyword_list_pull(pkd->keyword_view);
1036 keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
1037 string_list_free(keywords);
1038 bar_keyword_tree_sync(pkd);
1041 static void bar_pane_keywords_expand_checked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1043 PaneKeywordsData *pkd = data;
1044 pkd->expand_checked = !pkd->expand_checked;
1045 bar_keyword_tree_sync(pkd);
1048 static void bar_pane_keywords_collapse_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1050 PaneKeywordsData *pkd = data;
1051 pkd->collapse_unchecked = !pkd->collapse_unchecked;
1052 bar_keyword_tree_sync(pkd);
1055 static void bar_pane_keywords_hide_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1057 PaneKeywordsData *pkd = data;
1058 pkd->hide_unchecked = !pkd->hide_unchecked;
1059 bar_keyword_tree_sync(pkd);
1062 static void bar_pane_keywords_menu_popup(GtkWidget *widget, PaneKeywordsData *pkd, gint x, gint y)
1067 GtkTreeViewDropPosition pos;
1069 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1070 pkd->click_tpath = NULL;
1071 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(pkd->keyword_treeview), x, y, &pkd->click_tpath, &pos);
1073 menu = popup_menu_short_lived();
1075 if (pkd->click_tpath)
1082 GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1085 gtk_tree_model_get_iter(model, &iter, pkd->click_tpath);
1088 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
1089 FILTER_KEYWORD_COLUMN_MARK, &mark, -1);
1091 text = g_strdup_printf(_("Hide \"%s\""), name);
1092 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_hide_cb), pkd);
1095 submenu = gtk_menu_new();
1096 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
1098 text = g_strdup_printf(_("Mark %d"), i + 1);
1099 item = menu_item_add(submenu, text, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1100 g_object_set_data(G_OBJECT(item), "mark", GINT_TO_POINTER(i + 1));
1103 text = g_strdup_printf(_("Connect \"%s\" to mark"), name);
1104 item = menu_item_add(menu, text, NULL, NULL);
1105 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1108 menu_item_add_divider(menu);
1110 text = g_strdup_printf(_("Edit \"%s\""), name);
1111 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_edit_dialog_cb), pkd);
1113 text = g_strdup_printf(_("Delete \"%s\""), name);
1114 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_delete_cb), pkd);
1118 if (mark && mark[0])
1120 text = g_strdup_printf(_("Disconnect \"%s\" from mark %s"), name, mark);
1121 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1125 menu_item_add_divider(menu);
1132 menu_item_add(menu, _("Expand checked"), G_CALLBACK(bar_pane_keywords_expand_checked_cb), pkd);
1133 menu_item_add(menu, _("Collapse unchecked"), G_CALLBACK(bar_pane_keywords_collapse_unchecked_cb), pkd);
1134 menu_item_add(menu, _("Hide unchecked"), G_CALLBACK(bar_pane_keywords_hide_unchecked_cb), pkd);
1135 menu_item_add(menu, _("Show all"), G_CALLBACK(bar_pane_keywords_show_all_cb), pkd);
1137 submenu = gtk_menu_new();
1138 item = menu_item_add(menu, _("On any change"), NULL, NULL);
1139 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1141 menu_item_add_check(submenu, _("Expand checked"), pkd->expand_checked, G_CALLBACK(bar_pane_keywords_expand_checked_toggle_cb), pkd);
1142 menu_item_add_check(submenu, _("Collapse unchecked"), pkd->collapse_unchecked, G_CALLBACK(bar_pane_keywords_collapse_unchecked_toggle_cb), pkd);
1143 menu_item_add_check(submenu, _("Hide unchecked"), pkd->hide_unchecked, G_CALLBACK(bar_pane_keywords_hide_unchecked_toggle_cb), pkd);
1145 menu_item_add_divider(menu);
1147 menu_item_add_stock(menu, _("Add keyword"), GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_add_dialog_cb), pkd);
1149 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 0, GDK_CURRENT_TIME);
1153 static gboolean bar_pane_keywords_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
1155 PaneKeywordsData *pkd = data;
1156 if (bevent->button == MOUSE_BUTTON_RIGHT)
1158 bar_pane_keywords_menu_popup(widget, pkd, bevent->x, bevent->y);
1165 *-------------------------------------------------------------------
1167 *-------------------------------------------------------------------
1170 void bar_pane_keywords_close(GtkWidget *bar)
1172 PaneKeywordsData *pkd;
1174 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
1177 gtk_widget_destroy(pkd->widget);
1180 static void bar_pane_keywords_destroy(GtkWidget *widget, gpointer data)
1182 PaneKeywordsData *pkd = data;
1184 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1186 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
1188 file_data_unref(pkd->fd);
1195 GtkWidget *bar_pane_keywords_new(const gchar *title, const gchar *key, gboolean expanded)
1197 PaneKeywordsData *pkd;
1199 GtkWidget *scrolled;
1200 GtkTextBuffer *buffer;
1201 GtkTreeModel *store;
1202 GtkTreeViewColumn *column;
1203 GtkCellRenderer *renderer;
1205 pkd = g_new0(PaneKeywordsData, 1);
1207 pkd->pane.pane_set_fd = bar_pane_keywords_set_fd;
1208 pkd->pane.pane_event = bar_pane_keywords_event;
1209 pkd->pane.pane_write_config = bar_pane_keywords_write_config;
1210 pkd->pane.title = bar_pane_expander_title(title);
1212 pkd->pane.expanded = expanded;
1214 pkd->key = g_strdup(key);
1216 pkd->expand_checked = TRUE;
1218 hbox = gtk_hbox_new(FALSE, PREF_PAD_GAP);
1221 g_object_set_data(G_OBJECT(pkd->widget), "pane_data", pkd);
1222 g_signal_connect(G_OBJECT(pkd->widget), "destroy",
1223 G_CALLBACK(bar_pane_keywords_destroy), pkd);
1224 gtk_widget_show(hbox);
1226 scrolled = gtk_scrolled_window_new(NULL, NULL);
1227 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1228 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1229 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1230 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1231 gtk_widget_show(scrolled);
1233 pkd->keyword_view = gtk_text_view_new();
1234 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_view);
1235 g_signal_connect(G_OBJECT(pkd->keyword_view), "populate-popup",
1236 G_CALLBACK(bar_pane_keywords_populate_popup_cb), pkd);
1237 gtk_widget_show(pkd->keyword_view);
1239 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
1240 g_signal_connect(G_OBJECT(buffer), "changed",
1241 G_CALLBACK(bar_pane_keywords_changed), pkd);
1243 scrolled = gtk_scrolled_window_new(NULL, NULL);
1244 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1245 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1246 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1247 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1248 gtk_widget_show(scrolled);
1251 if (!keyword_tree) keyword_tree_new_default();
1253 store = gtk_tree_model_filter_new(GTK_TREE_MODEL(keyword_tree), NULL);
1255 gtk_tree_model_filter_set_modify_func(GTK_TREE_MODEL_FILTER(store),
1256 FILTER_KEYWORD_COLUMN_COUNT,
1257 filter_keyword_column_types,
1258 bar_pane_keywords_filter_modify,
1261 gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(store),
1262 bar_pane_keywords_filter_visible,
1266 pkd->keyword_treeview = gtk_tree_view_new_with_model(store);
1267 g_object_unref(store);
1269 gtk_widget_set_size_request(pkd->keyword_treeview, -1, 400);
1271 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(pkd->keyword_treeview), FALSE);
1273 // gtk_tree_view_set_search_column(GTK_TREE_VIEW(pkd->keyword_treeview), FILTER_KEYWORD_COLUMN_);
1275 column = gtk_tree_view_column_new();
1276 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
1278 renderer = gtk_cell_renderer_text_new();
1279 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1281 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_MARK);
1283 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1285 column = gtk_tree_view_column_new();
1286 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1287 renderer = gtk_cell_renderer_toggle_new();
1288 gtk_tree_view_column_pack_start(column, renderer, FALSE);
1289 gtk_tree_view_column_add_attribute(column, renderer, "active", FILTER_KEYWORD_COLUMN_TOGGLE);
1290 gtk_tree_view_column_add_attribute(column, renderer, "visible", FILTER_KEYWORD_COLUMN_IS_KEYWORD);
1291 g_signal_connect(G_OBJECT(renderer), "toggled",
1292 G_CALLBACK(bar_pane_keywords_keyword_toggle), pkd);
1294 renderer = gtk_cell_renderer_text_new();
1295 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1296 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_NAME);
1298 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1299 gtk_tree_view_set_expander_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1301 gtk_drag_source_set(pkd->keyword_treeview,
1302 GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
1303 bar_pane_keywords_drag_types, n_keywords_drag_types,
1304 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
1306 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_get",
1307 G_CALLBACK(bar_pane_keywords_dnd_get), pkd);
1309 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_begin",
1310 G_CALLBACK(bar_pane_keywords_dnd_begin), pkd);
1311 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_end",
1312 G_CALLBACK(bar_pane_keywords_dnd_end), pkd);
1314 gtk_drag_dest_set(pkd->keyword_treeview,
1315 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
1316 bar_pane_keywords_drop_types, n_keywords_drop_types,
1317 GDK_ACTION_COPY | GDK_ACTION_MOVE);
1319 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_received",
1320 G_CALLBACK(bar_pane_keywords_dnd_receive), pkd);
1322 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_motion",
1323 G_CALLBACK(bar_pane_keywords_dnd_motion), pkd);
1325 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "button_press_event",
1326 G_CALLBACK(bar_pane_keywords_menu_cb), pkd);
1328 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_treeview);
1329 gtk_widget_show(pkd->keyword_treeview);
1331 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
1336 GtkWidget *bar_pane_keywords_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
1338 gchar *title = g_strdup(_("NoName"));
1339 gchar *key = g_strdup(COMMENT_KEY);
1340 gboolean expanded = TRUE;
1342 while (*attribute_names)
1344 const gchar *option = *attribute_names++;
1345 const gchar *value = *attribute_values++;
1347 if (READ_CHAR_FULL("pane.title", title)) continue;
1348 if (READ_CHAR_FULL("key", key)) continue;
1349 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
1352 DEBUG_1("unknown attribute %s = %s", option, value);
1355 return bar_pane_keywords_new(title, key, expanded);
1358 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */