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"
32 static const gchar *keyword_favorite_defaults[] = {
44 static void bar_pane_keywords_keyword_update_all(void);
45 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data);
48 *-------------------------------------------------------------------
49 * keyword / comment utils
50 *-------------------------------------------------------------------
54 GList *keyword_list_pull(GtkWidget *text_widget)
59 text = text_widget_text_pull(text_widget);
60 list = string_to_keywords_list(text);
67 static void keyword_list_push(GtkWidget *textview, GList *list)
69 GtkTextBuffer *buffer;
70 GtkTextIter start, end;
72 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(textview));
73 gtk_text_buffer_get_bounds(buffer, &start, &end);
74 gtk_text_buffer_delete(buffer, &start, &end);
78 const gchar *word = list->data;
81 gtk_text_buffer_get_end_iter(buffer, &iter);
82 if (word) gtk_text_buffer_insert(buffer, &iter, word, -1);
83 gtk_text_buffer_get_end_iter(buffer, &iter);
84 gtk_text_buffer_insert(buffer, &iter, "\n", -1);
92 *-------------------------------------------------------------------
94 *-------------------------------------------------------------------
99 FILTER_KEYWORD_COLUMN_TOGGLE = 0,
100 FILTER_KEYWORD_COLUMN_MARK,
101 FILTER_KEYWORD_COLUMN_NAME,
102 FILTER_KEYWORD_COLUMN_IS_KEYWORD,
103 FILTER_KEYWORD_COLUMN_COUNT
106 static GType filter_keyword_column_types[] = {G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN};
108 typedef struct _PaneKeywordsData PaneKeywordsData;
109 struct _PaneKeywordsData
114 GtkWidget *keyword_view;
115 GtkWidget *keyword_treeview;
117 GtkTreePath *click_tpath;
119 gboolean expand_checked;
120 gboolean collapse_unchecked;
121 gboolean hide_unchecked;
127 typedef struct _ConfDialogData ConfDialogData;
128 struct _ConfDialogData
130 PaneKeywordsData *pkd;
131 GtkTreePath *click_tpath;
135 GtkWidget *edit_widget;
138 gboolean edit_existing;
141 static GList *bar_list = NULL;
144 static void bar_pane_keywords_write(PaneKeywordsData *pkd)
148 if (!pkd->fd) return;
150 list = keyword_list_pull(pkd->keyword_view);
152 metadata_write_list(pkd->fd, KEYWORD_KEY, list);
154 string_list_free(list);
157 gboolean bar_keyword_tree_expand_if_set_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
159 PaneKeywordsData *pkd = data;
162 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set, -1);
164 if (set && !gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
166 gtk_tree_view_expand_to_path(GTK_TREE_VIEW(pkd->keyword_treeview), path);
171 gboolean bar_keyword_tree_collapse_if_unset_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
173 PaneKeywordsData *pkd = data;
177 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set,
178 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
180 if (is_keyword && !set && gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
182 gtk_tree_view_collapse_row(GTK_TREE_VIEW(pkd->keyword_treeview), path);
187 static void bar_keyword_tree_sync(PaneKeywordsData *pkd)
191 GtkTreeModel *keyword_tree;
194 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
195 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
197 keywords = keyword_list_pull(pkd->keyword_view);
198 keyword_show_set_in(GTK_TREE_STORE(keyword_tree), model, keywords);
199 if (pkd->hide_unchecked) keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
200 string_list_free(keywords);
202 gtk_tree_model_filter_refilter(GTK_TREE_MODEL_FILTER(model));
204 if (pkd->expand_checked) gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
205 if (pkd->collapse_unchecked) gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
208 static void bar_pane_keywords_keyword_update_all(void)
215 PaneKeywordsData *pkd;
221 bar_keyword_tree_sync(pkd);
225 static void bar_pane_keywords_update(PaneKeywordsData *pkd)
227 GList *keywords = NULL;
228 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
230 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
232 keywords = metadata_read_list(pkd->fd, KEYWORD_KEY, METADATA_PLAIN);
233 keyword_list_push(pkd->keyword_view, keywords);
234 bar_keyword_tree_sync(pkd);
235 string_list_free(keywords);
237 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
241 void bar_pane_keywords_set_fd(GtkWidget *pane, FileData *fd)
243 PaneKeywordsData *pkd;
245 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
248 file_data_unref(pkd->fd);
249 pkd->fd = file_data_ref(fd);
251 bar_pane_keywords_update(pkd);
254 static void bar_pane_keywords_write_config(GtkWidget *pane, GString *outstr, gint indent)
256 PaneKeywordsData *pkd;
258 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
261 WRITE_STRING("<pane_keywords\n");
263 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(pkd->pane.title)));
264 WRITE_BOOL(*pkd, pane.expanded);
265 WRITE_CHAR(*pkd, key);
267 WRITE_STRING("/>\n");
270 gint bar_pane_keywords_event(GtkWidget *bar, GdkEvent *event)
272 PaneKeywordsData *pkd;
274 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
275 if (!pkd) return FALSE;
277 if (GTK_WIDGET_HAS_FOCUS(pkd->keyword_view)) return gtk_widget_event(pkd->keyword_view, event);
282 static void bar_pane_keywords_keyword_toggle(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
284 PaneKeywordsData *pkd = data;
290 GtkTreeIter child_iter;
291 GtkTreeModel *keyword_tree;
293 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
295 tpath = gtk_tree_path_new_from_string(path);
296 gtk_tree_model_get_iter(model, &iter, tpath);
297 gtk_tree_path_free(tpath);
299 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_TOGGLE, &active, -1);
303 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
304 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
306 list = keyword_list_pull(pkd->keyword_view);
308 keyword_tree_set(keyword_tree, &child_iter, &list);
310 keyword_tree_reset(keyword_tree, &child_iter, &list);
312 keyword_list_push(pkd->keyword_view, list);
313 string_list_free(list);
315 keyword_list_push triggers bar_pane_keywords_change which calls bar_keyword_tree_sync, no need to do it again
316 bar_keyword_tree_sync(pkd);
320 void bar_pane_keywords_filter_modify(GtkTreeModel *model, GtkTreeIter *iter, GValue *value, gint column, gpointer data)
322 PaneKeywordsData *pkd = data;
323 GtkTreeModel *keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
324 GtkTreeIter child_iter;
326 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, iter);
328 memset(value, 0, sizeof (GValue));
332 case FILTER_KEYWORD_COLUMN_TOGGLE:
334 GList *keywords = keyword_list_pull(pkd->keyword_view);
335 gboolean set = keyword_tree_is_set(keyword_tree, &child_iter, keywords);
336 string_list_free(keywords);
338 g_value_init(value, G_TYPE_BOOLEAN);
339 g_value_set_boolean(value, set);
342 case FILTER_KEYWORD_COLUMN_MARK:
343 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_MARK, value);
345 case FILTER_KEYWORD_COLUMN_NAME:
346 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_NAME, value);
348 case FILTER_KEYWORD_COLUMN_IS_KEYWORD:
349 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_IS_KEYWORD, value);
356 gboolean bar_pane_keywords_filter_visible(GtkTreeModel *keyword_tree, GtkTreeIter *iter, gpointer data)
358 GtkTreeModel *filter = data;
360 return !keyword_is_hidden_in(keyword_tree, iter, filter);
363 static void bar_pane_keywords_set_selection(PaneKeywordsData *pkd, gboolean append)
365 GList *keywords = NULL;
369 keywords = keyword_list_pull(pkd->keyword_view);
371 list = layout_selection_list(pkd->pane.lw);
375 FileData *fd = work->data;
380 metadata_append_list(fd, KEYWORD_KEY, keywords);
384 metadata_write_list(fd, KEYWORD_KEY, keywords);
389 string_list_free(keywords);
392 static void bar_pane_keywords_sel_add_cb(GtkWidget *button, gpointer data)
394 PaneKeywordsData *pkd = data;
396 bar_pane_keywords_set_selection(pkd, TRUE);
399 static void bar_pane_keywords_sel_replace_cb(GtkWidget *button, gpointer data)
401 PaneKeywordsData *pkd = data;
403 bar_pane_keywords_set_selection(pkd, FALSE);
406 static void bar_pane_keywords_populate_popup_cb(GtkTextView *textview, GtkMenu *menu, gpointer data)
408 PaneKeywordsData *pkd = data;
410 menu_item_add_divider(GTK_WIDGET(menu));
411 menu_item_add_stock(GTK_WIDGET(menu), _("Add keywords to selected files"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_keywords_sel_add_cb), pkd);
412 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);
416 static void bar_pane_keywords_notify_cb(FileData *fd, NotifyType type, gpointer data)
418 PaneKeywordsData *pkd = data;
419 if (fd == pkd->fd) bar_pane_keywords_update(pkd);
422 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data)
424 PaneKeywordsData *pkd = data;
426 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
427 bar_pane_keywords_write(pkd);
428 bar_keyword_tree_sync(pkd);
429 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
434 *-------------------------------------------------------------------
436 *-------------------------------------------------------------------
440 static GtkTargetEntry bar_pane_keywords_drag_types[] = {
441 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
442 { "text/plain", 0, TARGET_TEXT_PLAIN }
444 static gint n_keywords_drag_types = 2;
447 static GtkTargetEntry bar_pane_keywords_drop_types[] = {
448 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
449 { "text/plain", 0, TARGET_TEXT_PLAIN }
451 static gint n_keywords_drop_types = 2;
454 static void bar_pane_keywords_dnd_get(GtkWidget *tree_view, GdkDragContext *context,
455 GtkSelectionData *selection_data, guint info,
456 guint time, gpointer data)
460 GtkTreeIter child_iter;
461 GtkTreeModel *keyword_tree;
463 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
465 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
467 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
468 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
472 case TARGET_APP_KEYWORD_PATH:
474 GList *path = keyword_tree_get_path(keyword_tree, &child_iter);
475 gtk_selection_data_set(selection_data, selection_data->target,
476 8, (gpointer) &path, sizeof(path));
480 case TARGET_TEXT_PLAIN:
483 gchar *name = keyword_get_name(keyword_tree, &child_iter);
484 gtk_selection_data_set_text(selection_data, name, -1);
491 static void bar_pane_keywords_dnd_begin(GtkWidget *tree_view, GdkDragContext *context, gpointer data)
495 GtkTreeIter child_iter;
496 GtkTreeModel *keyword_tree;
499 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
501 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
503 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
504 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
506 name = keyword_get_name(keyword_tree, &child_iter);
508 dnd_set_drag_label(tree_view, context, name);
513 static void bar_pane_keywords_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
518 static gboolean bar_pane_keywords_dnd_can_move(GtkTreeModel *keyword_tree, GtkTreeIter *src_kw_iter, GtkTreeIter *dest_kw_iter)
523 if (dest_kw_iter && keyword_same_parent(keyword_tree, src_kw_iter, dest_kw_iter))
525 return TRUE; /* reordering of siblings is ok */
527 if (!dest_kw_iter && !gtk_tree_model_iter_parent(keyword_tree, &parent, src_kw_iter))
529 return TRUE; /* reordering of top-level siblings is ok */
532 src_name = keyword_get_name(keyword_tree, src_kw_iter);
533 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, src_name, FALSE))
542 static gboolean bar_pane_keywords_dnd_skip_existing(GtkTreeModel *keyword_tree, GtkTreeIter *dest_kw_iter, GList **keywords)
544 /* we have to find at least one keyword that does not already exist as a sibling of dest_kw_iter */
545 GList *work = *keywords;
548 gchar *keyword = work->data;
549 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, keyword, FALSE))
551 GList *next = work->next;
553 *keywords = g_list_delete_link(*keywords, work);
564 static void bar_pane_keywords_dnd_receive(GtkWidget *tree_view, GdkDragContext *context,
566 GtkSelectionData *selection_data, guint info,
567 guint time, gpointer data)
569 PaneKeywordsData *pkd = data;
570 GtkTreePath *tpath = NULL;
571 GtkTreeViewDropPosition pos;
574 GtkTreeModel *keyword_tree;
575 gboolean src_valid = FALSE;
576 GList *new_keywords = NULL;
579 /* iterators for keyword_tree */
580 GtkTreeIter src_kw_iter;
581 GtkTreeIter dest_kw_iter;
582 GtkTreeIter new_kw_iter;
584 g_signal_stop_emission_by_name(tree_view, "drag_data_received");
586 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), NULL, pos);
588 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
589 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
591 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
596 case TARGET_APP_KEYWORD_PATH:
598 GList *path = *(gpointer *)selection_data->data;
599 src_valid = keyword_tree_get_iter(keyword_tree, &src_kw_iter, path);
600 string_list_free(path);
604 new_keywords = string_to_keywords_list((gchar *)selection_data->data);
610 GtkTreeIter dest_iter;
611 gtk_tree_model_get_iter(model, &dest_iter, tpath);
612 gtk_tree_path_free(tpath);
613 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &dest_kw_iter, &dest_iter);
615 if (src_valid && gtk_tree_store_is_ancestor(GTK_TREE_STORE(keyword_tree), &src_kw_iter, &dest_kw_iter))
617 /* can't move to it's own child */
621 if (src_valid && keyword_compare(keyword_tree, &src_kw_iter, &dest_kw_iter) == 0)
623 /* can't move to itself */
627 if ((pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE || pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER) &&
628 !gtk_tree_model_iter_has_child(keyword_tree, &dest_kw_iter))
630 /* the node has no children, all keywords can be added */
631 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &dest_kw_iter);
635 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, &dest_kw_iter))
637 /* the keyword can't be moved if the same name already exist */
640 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, &dest_kw_iter, &new_keywords))
642 /* the keywords can't be added if the same name already exist */
648 case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
649 case GTK_TREE_VIEW_DROP_BEFORE:
650 gtk_tree_store_insert_before(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
652 case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
653 case GTK_TREE_VIEW_DROP_AFTER:
654 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
662 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, NULL))
664 /* the keyword can't be moved if the same name already exist */
667 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, NULL, &new_keywords))
669 /* the keywords can't be added if the same name already exist */
672 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL);
678 keyword_move_recursive(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &src_kw_iter);
684 gchar *keyword = work->data;
685 keyword_set(GTK_TREE_STORE(keyword_tree), &new_kw_iter, keyword, TRUE);
691 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &new_kw_iter);
695 string_list_free(new_keywords);
696 bar_keyword_tree_sync(pkd);
699 static gint bar_pane_keywords_dnd_motion(GtkWidget *tree_view, GdkDragContext *context,
700 gint x, gint y, guint time, gpointer data)
702 GtkTreePath *tpath = NULL;
703 GtkTreeViewDropPosition pos;
704 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
708 GtkTreeIter dest_iter;
709 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
710 gtk_tree_model_get_iter(model, &dest_iter, tpath);
711 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE && gtk_tree_model_iter_has_child(model, &dest_iter))
712 pos = GTK_TREE_VIEW_DROP_BEFORE;
714 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER && gtk_tree_model_iter_has_child(model, &dest_iter))
715 pos = GTK_TREE_VIEW_DROP_AFTER;
718 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), tpath, pos);
719 gtk_tree_path_free(tpath);
721 if (tree_view == gtk_drag_get_source_widget(context))
722 gdk_drag_status(context, GDK_ACTION_MOVE, time);
724 gdk_drag_status(context, GDK_ACTION_COPY, time);
730 *-------------------------------------------------------------------
732 *-------------------------------------------------------------------
735 static void bar_pane_keywords_edit_destroy_cb(GtkWidget *widget, gpointer data)
737 ConfDialogData *cdd = data;
738 gtk_tree_path_free(cdd->click_tpath);
743 static void bar_pane_keywords_edit_cancel_cb(GenericDialog *gd, gpointer data)
748 static void bar_pane_keywords_edit_ok_cb(GenericDialog *gd, gpointer data)
750 ConfDialogData *cdd = data;
751 PaneKeywordsData *pkd = cdd->pkd;
754 GtkTreeModel *keyword_tree;
757 gboolean have_dest = FALSE;
761 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
762 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
764 if (cdd->click_tpath)
767 if (gtk_tree_model_get_iter(model, &iter, cdd->click_tpath))
769 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
774 if (cdd->edit_existing && !have_dest) return;
776 keywords = keyword_list_pull(cdd->edit_widget);
778 if (cdd->edit_existing)
780 if (keywords && keywords->data && /* there should be one keyword */
781 !keyword_exists(keyword_tree, NULL, &kw_iter, keywords->data, TRUE))
783 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, keywords->data, cdd->is_keyword);
788 GList *work = keywords;
793 if (keyword_exists(keyword_tree, NULL, have_dest ? &kw_iter : NULL, work->data, FALSE))
800 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &kw_iter);
804 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &add, NULL);
808 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, work->data, cdd->is_keyword);
812 string_list_free(keywords);
815 static void bar_pane_keywords_conf_set_helper(GtkWidget *widget, gpointer data)
817 ConfDialogData *cdd = data;
818 cdd->is_keyword = FALSE;
821 static void bar_pane_keywords_conf_set_kw(GtkWidget *widget, gpointer data)
823 ConfDialogData *cdd = data;
824 cdd->is_keyword = TRUE;
829 static void bar_pane_keywords_edit_dialog(PaneKeywordsData *pkd, gboolean edit_existing)
838 gboolean is_keyword = TRUE;
841 if (edit_existing && pkd->click_tpath)
845 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
847 if (gtk_tree_model_get_iter(model, &iter, pkd->click_tpath))
849 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
850 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
858 if (edit_existing && !name) return;
861 cdd = g_new0(ConfDialogData, 1);
863 cdd->click_tpath = pkd->click_tpath;
864 pkd->click_tpath = NULL;
865 cdd->is_keyword = is_keyword;
866 cdd->edit_existing = edit_existing;
868 cdd->gd = gd = generic_dialog_new(name ? _("Edit keyword") : _("Add keywords"), "keyword_edit",
870 bar_pane_keywords_edit_cancel_cb, cdd);
871 g_signal_connect(G_OBJECT(gd->dialog), "destroy",
872 G_CALLBACK(bar_pane_keywords_edit_destroy_cb), cdd);
875 generic_dialog_add_message(gd, NULL, name ? _("Configure keyword") : _("Add keyword"), NULL);
877 generic_dialog_add_button(gd, GTK_STOCK_OK, NULL,
878 bar_pane_keywords_edit_ok_cb, TRUE);
880 table = pref_table_new(gd->vbox, 3, 1, FALSE, TRUE);
881 pref_table_label(table, 0, 0, _("Keyword:"), 1.0);
882 cdd->edit_widget = gtk_entry_new();
883 gtk_widget_set_size_request(cdd->edit_widget, 300, -1);
884 if (name) gtk_entry_set_text(GTK_ENTRY(cdd->edit_widget), name);
885 gtk_table_attach_defaults(GTK_TABLE(table), cdd->edit_widget, 1, 2, 0, 1);
886 /* here could eventually be a text view instead of entry */
887 generic_dialog_attach_default(gd, cdd->edit_widget);
888 gtk_widget_show(cdd->edit_widget);
890 group = pref_group_new(gd->vbox, FALSE, _("Keyword type:"), GTK_ORIENTATION_VERTICAL);
892 button = pref_radiobutton_new(group, NULL, _("Active keyword"),
894 G_CALLBACK(bar_pane_keywords_conf_set_kw), cdd);
895 button = pref_radiobutton_new(group, button, _("Helper"),
897 G_CALLBACK(bar_pane_keywords_conf_set_helper), cdd);
900 gtk_widget_show(gd->dialog);
907 *-------------------------------------------------------------------
909 *-------------------------------------------------------------------
912 static void bar_pane_keywords_edit_dialog_cb(GtkWidget *menu_widget, gpointer data)
914 PaneKeywordsData *pkd = data;
915 bar_pane_keywords_edit_dialog(pkd, TRUE);
918 static void bar_pane_keywords_add_dialog_cb(GtkWidget *menu_widget, gpointer data)
920 PaneKeywordsData *pkd = data;
921 bar_pane_keywords_edit_dialog(pkd, FALSE);
924 static void bar_pane_keywords_connect_mark_cb(GtkWidget *menu_widget, gpointer data)
926 PaneKeywordsData *pkd = data;
931 GtkTreeModel *keyword_tree;
934 gint mark = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menu_widget), "mark")) - 1;
936 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
937 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
939 if (!pkd->click_tpath) return;
940 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
942 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
944 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
946 meta_data_connect_mark_with_keyword(keyword_tree, &kw_iter, mark);
948 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
949 // bar_pane_keywords_update(pkd);
953 static void bar_pane_keywords_delete_cb(GtkWidget *menu_widget, gpointer data)
955 PaneKeywordsData *pkd = data;
959 GtkTreeModel *keyword_tree;
962 if (!pkd->click_tpath) return;
964 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
965 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
967 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
968 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
970 keyword_delete(GTK_TREE_STORE(keyword_tree), &kw_iter);
973 static void bar_pane_keywords_hide_cb(GtkWidget *menu_widget, gpointer data)
975 PaneKeywordsData *pkd = data;
979 GtkTreeModel *keyword_tree;
982 if (!pkd->click_tpath) return;
984 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
985 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
987 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
988 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
990 keyword_hide_in(GTK_TREE_STORE(keyword_tree), &kw_iter, model);
993 static void bar_pane_keywords_show_all_cb(GtkWidget *menu_widget, gpointer data)
995 PaneKeywordsData *pkd = data;
998 GtkTreeModel *keyword_tree;
1000 pkd->hide_unchecked = FALSE;
1002 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1003 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1005 keyword_show_all_in(GTK_TREE_STORE(keyword_tree), model);
1006 bar_keyword_tree_sync(pkd);
1009 static void bar_pane_keywords_expand_checked_cb(GtkWidget *menu_widget, gpointer data)
1011 PaneKeywordsData *pkd = data;
1012 GtkTreeModel *model;
1014 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1015 gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
1018 static void bar_pane_keywords_collapse_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1020 PaneKeywordsData *pkd = data;
1021 GtkTreeModel *model;
1023 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1024 gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
1027 static void bar_pane_keywords_hide_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1029 PaneKeywordsData *pkd = data;
1030 GtkTreeModel *model;
1032 GtkTreeModel *keyword_tree;
1035 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1036 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1038 keywords = keyword_list_pull(pkd->keyword_view);
1039 keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
1040 string_list_free(keywords);
1041 bar_keyword_tree_sync(pkd);
1044 static void bar_pane_keywords_expand_checked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1046 PaneKeywordsData *pkd = data;
1047 pkd->expand_checked = !pkd->expand_checked;
1048 bar_keyword_tree_sync(pkd);
1051 static void bar_pane_keywords_collapse_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1053 PaneKeywordsData *pkd = data;
1054 pkd->collapse_unchecked = !pkd->collapse_unchecked;
1055 bar_keyword_tree_sync(pkd);
1058 static void bar_pane_keywords_hide_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1060 PaneKeywordsData *pkd = data;
1061 pkd->hide_unchecked = !pkd->hide_unchecked;
1062 bar_keyword_tree_sync(pkd);
1065 static void bar_pane_keywords_menu_popup(GtkWidget *widget, PaneKeywordsData *pkd, gint x, gint y)
1070 GtkTreeViewDropPosition pos;
1072 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1073 pkd->click_tpath = NULL;
1074 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(pkd->keyword_treeview), x, y, &pkd->click_tpath, &pos);
1076 menu = popup_menu_short_lived();
1078 if (pkd->click_tpath)
1085 GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1088 gtk_tree_model_get_iter(model, &iter, pkd->click_tpath);
1091 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
1092 FILTER_KEYWORD_COLUMN_MARK, &mark, -1);
1094 text = g_strdup_printf(_("Hide \"%s\""), name);
1095 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_hide_cb), pkd);
1098 submenu = gtk_menu_new();
1099 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
1101 text = g_strdup_printf(_("Mark %d"), i + 1);
1102 item = menu_item_add(submenu, text, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1103 g_object_set_data(G_OBJECT(item), "mark", GINT_TO_POINTER(i + 1));
1106 text = g_strdup_printf(_("Connect \"%s\" to mark"), name);
1107 item = menu_item_add(menu, text, NULL, NULL);
1108 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1111 menu_item_add_divider(menu);
1113 text = g_strdup_printf(_("Edit \"%s\""), name);
1114 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_edit_dialog_cb), pkd);
1116 text = g_strdup_printf(_("Delete \"%s\""), name);
1117 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_delete_cb), pkd);
1121 if (mark && mark[0])
1123 text = g_strdup_printf(_("Disconnect \"%s\" from mark %s"), name, mark);
1124 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1128 menu_item_add_divider(menu);
1135 menu_item_add(menu, _("Expand checked"), G_CALLBACK(bar_pane_keywords_expand_checked_cb), pkd);
1136 menu_item_add(menu, _("Collapse unchecked"), G_CALLBACK(bar_pane_keywords_collapse_unchecked_cb), pkd);
1137 menu_item_add(menu, _("Hide unchecked"), G_CALLBACK(bar_pane_keywords_hide_unchecked_cb), pkd);
1138 menu_item_add(menu, _("Show all"), G_CALLBACK(bar_pane_keywords_show_all_cb), pkd);
1140 submenu = gtk_menu_new();
1141 item = menu_item_add(menu, _("On any change"), NULL, NULL);
1142 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1144 menu_item_add_check(submenu, _("Expand checked"), pkd->expand_checked, G_CALLBACK(bar_pane_keywords_expand_checked_toggle_cb), pkd);
1145 menu_item_add_check(submenu, _("Collapse unchecked"), pkd->collapse_unchecked, G_CALLBACK(bar_pane_keywords_collapse_unchecked_toggle_cb), pkd);
1146 menu_item_add_check(submenu, _("Hide unchecked"), pkd->hide_unchecked, G_CALLBACK(bar_pane_keywords_hide_unchecked_toggle_cb), pkd);
1148 menu_item_add_divider(menu);
1150 menu_item_add_stock(menu, _("Add keyword"), GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_add_dialog_cb), pkd);
1152 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 0, GDK_CURRENT_TIME);
1156 static gboolean bar_pane_keywords_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
1158 PaneKeywordsData *pkd = data;
1159 if (bevent->button == MOUSE_BUTTON_RIGHT)
1161 bar_pane_keywords_menu_popup(widget, pkd, bevent->x, bevent->y);
1168 *-------------------------------------------------------------------
1170 *-------------------------------------------------------------------
1173 void bar_pane_keywords_close(GtkWidget *bar)
1175 PaneKeywordsData *pkd;
1177 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
1180 gtk_widget_destroy(pkd->widget);
1183 static void bar_pane_keywords_destroy(GtkWidget *widget, gpointer data)
1185 PaneKeywordsData *pkd = data;
1187 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1189 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
1191 file_data_unref(pkd->fd);
1198 GtkWidget *bar_pane_keywords_new(const gchar *title, const gchar *key, gboolean expanded)
1200 PaneKeywordsData *pkd;
1202 GtkWidget *scrolled;
1203 GtkTextBuffer *buffer;
1204 GtkTreeModel *store;
1205 GtkTreeViewColumn *column;
1206 GtkCellRenderer *renderer;
1208 pkd = g_new0(PaneKeywordsData, 1);
1210 pkd->pane.pane_set_fd = bar_pane_keywords_set_fd;
1211 pkd->pane.pane_event = bar_pane_keywords_event;
1212 pkd->pane.pane_write_config = bar_pane_keywords_write_config;
1213 pkd->pane.title = bar_pane_expander_title(title);
1215 pkd->pane.expanded = expanded;
1217 pkd->key = g_strdup(key);
1219 pkd->expand_checked = TRUE;
1221 hbox = gtk_hbox_new(FALSE, PREF_PAD_GAP);
1224 g_object_set_data(G_OBJECT(pkd->widget), "pane_data", pkd);
1225 g_signal_connect(G_OBJECT(pkd->widget), "destroy",
1226 G_CALLBACK(bar_pane_keywords_destroy), pkd);
1227 gtk_widget_show(hbox);
1229 scrolled = gtk_scrolled_window_new(NULL, NULL);
1230 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1231 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1232 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1233 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1234 gtk_widget_show(scrolled);
1236 pkd->keyword_view = gtk_text_view_new();
1237 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_view);
1238 g_signal_connect(G_OBJECT(pkd->keyword_view), "populate-popup",
1239 G_CALLBACK(bar_pane_keywords_populate_popup_cb), pkd);
1240 gtk_widget_show(pkd->keyword_view);
1242 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
1243 g_signal_connect(G_OBJECT(buffer), "changed",
1244 G_CALLBACK(bar_pane_keywords_changed), pkd);
1246 scrolled = gtk_scrolled_window_new(NULL, NULL);
1247 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1248 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1249 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1250 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1251 gtk_widget_show(scrolled);
1254 if (!keyword_tree) keyword_tree_new_default();
1256 store = gtk_tree_model_filter_new(GTK_TREE_MODEL(keyword_tree), NULL);
1258 gtk_tree_model_filter_set_modify_func(GTK_TREE_MODEL_FILTER(store),
1259 FILTER_KEYWORD_COLUMN_COUNT,
1260 filter_keyword_column_types,
1261 bar_pane_keywords_filter_modify,
1264 gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(store),
1265 bar_pane_keywords_filter_visible,
1269 pkd->keyword_treeview = gtk_tree_view_new_with_model(store);
1270 g_object_unref(store);
1272 gtk_widget_set_size_request(pkd->keyword_treeview, -1, 400);
1274 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(pkd->keyword_treeview), FALSE);
1276 // gtk_tree_view_set_search_column(GTK_TREE_VIEW(pkd->keyword_treeview), FILTER_KEYWORD_COLUMN_);
1278 column = gtk_tree_view_column_new();
1279 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
1281 renderer = gtk_cell_renderer_text_new();
1282 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1284 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_MARK);
1286 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1288 column = gtk_tree_view_column_new();
1289 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1290 renderer = gtk_cell_renderer_toggle_new();
1291 gtk_tree_view_column_pack_start(column, renderer, FALSE);
1292 gtk_tree_view_column_add_attribute(column, renderer, "active", FILTER_KEYWORD_COLUMN_TOGGLE);
1293 gtk_tree_view_column_add_attribute(column, renderer, "visible", FILTER_KEYWORD_COLUMN_IS_KEYWORD);
1294 g_signal_connect(G_OBJECT(renderer), "toggled",
1295 G_CALLBACK(bar_pane_keywords_keyword_toggle), pkd);
1297 renderer = gtk_cell_renderer_text_new();
1298 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1299 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_NAME);
1301 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1302 gtk_tree_view_set_expander_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1304 gtk_drag_source_set(pkd->keyword_treeview,
1305 GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
1306 bar_pane_keywords_drag_types, n_keywords_drag_types,
1307 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
1309 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_get",
1310 G_CALLBACK(bar_pane_keywords_dnd_get), pkd);
1312 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_begin",
1313 G_CALLBACK(bar_pane_keywords_dnd_begin), pkd);
1314 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_end",
1315 G_CALLBACK(bar_pane_keywords_dnd_end), pkd);
1317 gtk_drag_dest_set(pkd->keyword_treeview,
1318 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
1319 bar_pane_keywords_drop_types, n_keywords_drop_types,
1320 GDK_ACTION_COPY | GDK_ACTION_MOVE);
1322 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_received",
1323 G_CALLBACK(bar_pane_keywords_dnd_receive), pkd);
1325 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_motion",
1326 G_CALLBACK(bar_pane_keywords_dnd_motion), pkd);
1328 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "button_press_event",
1329 G_CALLBACK(bar_pane_keywords_menu_cb), pkd);
1331 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_treeview);
1332 gtk_widget_show(pkd->keyword_treeview);
1334 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
1339 GtkWidget *bar_pane_keywords_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
1341 gchar *title = g_strdup(_("NoName"));
1342 gchar *key = g_strdup(COMMENT_KEY);
1343 gboolean expanded = TRUE;
1345 while (*attribute_names)
1347 const gchar *option = *attribute_names++;
1348 const gchar *value = *attribute_values++;
1350 if (READ_CHAR_FULL("pane.title", title)) continue;
1351 if (READ_CHAR_FULL("key", key)) continue;
1352 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
1355 DEBUG_1("unknown attribute %s = %s", option, value);
1358 return bar_pane_keywords_new(title, key, expanded);
1361 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */