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;
123 typedef struct _ConfDialogData ConfDialogData;
124 struct _ConfDialogData
126 PaneKeywordsData *pkd;
127 GtkTreePath *click_tpath;
131 GtkWidget *edit_widget;
134 gboolean edit_existing;
137 static GList *bar_list = NULL;
140 static void bar_pane_keywords_write(PaneKeywordsData *pkd)
144 if (!pkd->fd) return;
146 list = keyword_list_pull(pkd->keyword_view);
148 metadata_write_list(pkd->fd, KEYWORD_KEY, list);
150 string_list_free(list);
153 gboolean bar_keyword_tree_expand_if_set(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
155 PaneKeywordsData *pkd = data;
158 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set, -1);
160 if (set && !gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
162 gtk_tree_view_expand_to_path(GTK_TREE_VIEW(pkd->keyword_treeview), path);
167 static void bar_keyword_tree_sync(PaneKeywordsData *pkd)
169 GtkTreeModelFilter *store;
171 store = GTK_TREE_MODEL_FILTER(gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview)));
173 gtk_tree_model_filter_refilter(store);
174 gtk_tree_model_foreach(GTK_TREE_MODEL(store), bar_keyword_tree_expand_if_set, pkd);
178 static void bar_pane_keywords_keyword_update_all(void)
185 PaneKeywordsData *pkd;
191 bar_keyword_tree_sync(pkd);
195 static void bar_pane_keywords_update(PaneKeywordsData *pkd)
197 GList *keywords = NULL;
198 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
200 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
202 keywords = metadata_read_list(pkd->fd, KEYWORD_KEY, METADATA_PLAIN);
203 keyword_list_push(pkd->keyword_view, keywords);
204 bar_keyword_tree_sync(pkd);
205 string_list_free(keywords);
207 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
211 void bar_pane_keywords_set_fd(GtkWidget *pane, FileData *fd)
213 PaneKeywordsData *pkd;
215 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
218 file_data_unref(pkd->fd);
219 pkd->fd = file_data_ref(fd);
221 bar_pane_keywords_update(pkd);
224 static void bar_pane_keywords_write_config(GtkWidget *pane, GString *outstr, gint indent)
226 PaneKeywordsData *pkd;
228 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
231 WRITE_STRING("<pane_keywords\n");
233 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(pkd->pane.title)));
234 WRITE_BOOL(*pkd, pane.expanded);
235 WRITE_CHAR(*pkd, key);
237 WRITE_STRING("/>\n");
240 gint bar_pane_keywords_event(GtkWidget *bar, GdkEvent *event)
242 PaneKeywordsData *pkd;
244 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
245 if (!pkd) return FALSE;
247 if (GTK_WIDGET_HAS_FOCUS(pkd->keyword_view)) return gtk_widget_event(pkd->keyword_view, event);
252 static void bar_pane_keywords_keyword_toggle(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
254 PaneKeywordsData *pkd = data;
260 GtkTreeIter child_iter;
261 GtkTreeModel *keyword_tree;
263 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
265 tpath = gtk_tree_path_new_from_string(path);
266 gtk_tree_model_get_iter(model, &iter, tpath);
267 gtk_tree_path_free(tpath);
269 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_TOGGLE, &active, -1);
273 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
274 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
276 list = keyword_list_pull(pkd->keyword_view);
278 keyword_tree_set(keyword_tree, &child_iter, &list);
280 keyword_tree_reset(keyword_tree, &child_iter, &list);
282 keyword_list_push(pkd->keyword_view, list);
283 string_list_free(list);
285 keyword_list_push triggers bar_pane_keywords_change which calls bar_keyword_tree_sync, no need to do it again
286 bar_keyword_tree_sync(pkd);
290 void bar_pane_keywords_filter_modify(GtkTreeModel *model, GtkTreeIter *iter, GValue *value, gint column, gpointer data)
292 PaneKeywordsData *pkd = data;
293 GtkTreeModel *keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
294 GtkTreeIter child_iter;
296 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, iter);
298 memset(value, 0, sizeof (GValue));
302 case FILTER_KEYWORD_COLUMN_TOGGLE:
304 GList *keywords = keyword_list_pull(pkd->keyword_view);
305 gboolean set = keyword_tree_is_set(keyword_tree, &child_iter, keywords);
306 string_list_free(keywords);
308 g_value_init(value, G_TYPE_BOOLEAN);
309 g_value_set_boolean(value, set);
312 case FILTER_KEYWORD_COLUMN_MARK:
313 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_MARK, value);
315 case FILTER_KEYWORD_COLUMN_NAME:
316 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_NAME, value);
318 case FILTER_KEYWORD_COLUMN_IS_KEYWORD:
319 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_IS_KEYWORD, value);
326 static void bar_pane_keywords_set_selection(PaneKeywordsData *pkd, gboolean append)
328 GList *keywords = NULL;
332 keywords = keyword_list_pull(pkd->keyword_view);
334 list = layout_selection_list(pkd->pane.lw);
338 FileData *fd = work->data;
343 metadata_append_list(fd, KEYWORD_KEY, keywords);
347 metadata_write_list(fd, KEYWORD_KEY, keywords);
352 string_list_free(keywords);
355 static void bar_pane_keywords_sel_add_cb(GtkWidget *button, gpointer data)
357 PaneKeywordsData *pkd = data;
359 bar_pane_keywords_set_selection(pkd, TRUE);
362 static void bar_pane_keywords_sel_replace_cb(GtkWidget *button, gpointer data)
364 PaneKeywordsData *pkd = data;
366 bar_pane_keywords_set_selection(pkd, FALSE);
369 static void bar_pane_keywords_populate_popup_cb(GtkTextView *textview, GtkMenu *menu, gpointer data)
371 PaneKeywordsData *pkd = data;
373 menu_item_add_divider(GTK_WIDGET(menu));
374 menu_item_add_stock(GTK_WIDGET(menu), _("Add keywords to selected files"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_keywords_sel_add_cb), pkd);
375 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);
379 static void bar_pane_keywords_notify_cb(FileData *fd, NotifyType type, gpointer data)
381 PaneKeywordsData *pkd = data;
382 if (fd == pkd->fd) bar_pane_keywords_update(pkd);
385 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data)
387 PaneKeywordsData *pkd = data;
389 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
390 bar_pane_keywords_write(pkd);
391 bar_keyword_tree_sync(pkd);
392 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
397 *-------------------------------------------------------------------
399 *-------------------------------------------------------------------
403 static GtkTargetEntry bar_pane_keywords_drag_types[] = {
404 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
405 { "text/plain", 0, TARGET_TEXT_PLAIN }
407 static gint n_keywords_drag_types = 2;
410 static GtkTargetEntry bar_pane_keywords_drop_types[] = {
411 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
412 { "text/plain", 0, TARGET_TEXT_PLAIN }
414 static gint n_keywords_drop_types = 2;
417 static void bar_pane_keywords_dnd_get(GtkWidget *tree_view, GdkDragContext *context,
418 GtkSelectionData *selection_data, guint info,
419 guint time, gpointer data)
423 GtkTreeIter child_iter;
424 GtkTreeModel *keyword_tree;
426 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
428 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
430 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
431 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
435 case TARGET_APP_KEYWORD_PATH:
437 GList *path = keyword_tree_get_path(keyword_tree, &child_iter);
438 gtk_selection_data_set(selection_data, selection_data->target,
439 8, (gpointer) &path, sizeof(path));
443 case TARGET_TEXT_PLAIN:
446 gchar *name = keyword_get_name(keyword_tree, &child_iter);
447 gtk_selection_data_set_text(selection_data, name, -1);
454 static void bar_pane_keywords_dnd_begin(GtkWidget *tree_view, GdkDragContext *context, gpointer data)
458 GtkTreeIter child_iter;
459 GtkTreeModel *keyword_tree;
462 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
464 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
466 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
467 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
469 name = keyword_get_name(keyword_tree, &child_iter);
471 dnd_set_drag_label(tree_view, context, name);
476 static void bar_pane_keywords_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
480 static void bar_pane_keywords_dnd_receive(GtkWidget *tree_view, GdkDragContext *context,
482 GtkSelectionData *selection_data, guint info,
483 guint time, gpointer data)
485 PaneKeywordsData *pkd = data;
486 GtkTreePath *tpath = NULL;
487 GtkTreeViewDropPosition pos;
490 GtkTreeModel *keyword_tree;
491 gboolean src_valid = FALSE;
492 GList *new_keywords = NULL;
495 /* iterators for keyword_tree */
496 GtkTreeIter src_kw_iter;
497 GtkTreeIter dest_kw_iter;
498 GtkTreeIter new_kw_iter;
500 g_signal_stop_emission_by_name(tree_view, "drag_data_received");
502 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), NULL, pos);
504 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
505 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
507 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
512 case TARGET_APP_KEYWORD_PATH:
514 GList *path = *(gpointer *)selection_data->data;
515 src_valid = keyword_tree_get_iter(keyword_tree, &src_kw_iter, path);
516 string_list_free(path);
520 new_keywords = string_to_keywords_list((gchar *)selection_data->data);
526 GtkTreeIter dest_iter;
527 gtk_tree_model_get_iter(model, &dest_iter, tpath);
528 gtk_tree_path_free(tpath);
529 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &dest_kw_iter, &dest_iter);
531 if (src_valid && gtk_tree_store_is_ancestor(GTK_TREE_STORE(keyword_tree), &src_kw_iter, &dest_kw_iter))
533 /* can't move to it's own child */
537 if (src_valid && keyword_compare(keyword_tree, &src_kw_iter, &dest_kw_iter) == 0)
539 /* can't move to itself */
543 if ((pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE || pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER) &&
544 !gtk_tree_model_iter_has_child(keyword_tree, &dest_kw_iter))
546 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &dest_kw_iter);
552 case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
553 case GTK_TREE_VIEW_DROP_BEFORE:
554 gtk_tree_store_insert_before(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
556 case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
557 case GTK_TREE_VIEW_DROP_AFTER:
558 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
565 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL);
571 keyword_move_recursive(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &src_kw_iter);
577 keyword_set(GTK_TREE_STORE(keyword_tree), &new_kw_iter, work->data, TRUE);
582 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &new_kw_iter);
586 string_list_free(new_keywords);
587 bar_keyword_tree_sync(pkd);
590 static gint bar_pane_keywords_dnd_motion(GtkWidget *tree_view, GdkDragContext *context,
591 gint x, gint y, guint time, gpointer data)
593 GtkTreePath *tpath = NULL;
594 GtkTreeViewDropPosition pos;
595 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
599 GtkTreeIter dest_iter;
600 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
601 gtk_tree_model_get_iter(model, &dest_iter, tpath);
602 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE && gtk_tree_model_iter_has_child(model, &dest_iter))
603 pos = GTK_TREE_VIEW_DROP_BEFORE;
605 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER && gtk_tree_model_iter_has_child(model, &dest_iter))
606 pos = GTK_TREE_VIEW_DROP_AFTER;
609 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), tpath, pos);
610 gtk_tree_path_free(tpath);
612 if (tree_view == gtk_drag_get_source_widget(context))
613 gdk_drag_status(context, GDK_ACTION_MOVE, time);
615 gdk_drag_status(context, GDK_ACTION_COPY, time);
621 *-------------------------------------------------------------------
623 *-------------------------------------------------------------------
626 static void bar_pane_keywords_edit_destroy_cb(GtkWidget *widget, gpointer data)
628 ConfDialogData *cdd = data;
629 gtk_tree_path_free(cdd->click_tpath);
634 static void bar_pane_keywords_edit_cancel_cb(GenericDialog *gd, gpointer data)
639 static void bar_pane_keywords_edit_ok_cb(GenericDialog *gd, gpointer data)
641 ConfDialogData *cdd = data;
642 PaneKeywordsData *pkd = cdd->pkd;
645 GtkTreeModel *keyword_tree;
648 gboolean have_dest = FALSE;
652 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
653 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
655 if (cdd->click_tpath)
658 if (gtk_tree_model_get_iter(model, &iter, cdd->click_tpath))
660 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
665 if (cdd->edit_existing && !have_dest) return;
667 keywords = keyword_list_pull(cdd->edit_widget);
669 if (cdd->edit_existing)
671 if (keywords && keywords->data) /* there should be one keyword */
673 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, keywords->data, cdd->is_keyword);
678 GList *work = keywords;
685 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &kw_iter);
689 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &add, NULL);
693 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, work->data, cdd->is_keyword);
697 string_list_free(keywords);
700 static void bar_pane_keywords_conf_set_helper(GtkWidget *widget, gpointer data)
702 ConfDialogData *cdd = data;
703 cdd->is_keyword = FALSE;
706 static void bar_pane_keywords_conf_set_kw(GtkWidget *widget, gpointer data)
708 ConfDialogData *cdd = data;
709 cdd->is_keyword = TRUE;
714 static void bar_pane_keywords_edit_dialog(PaneKeywordsData *pkd, gboolean edit_existing)
723 gboolean is_keyword = TRUE;
726 if (edit_existing && pkd->click_tpath)
730 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
732 if (gtk_tree_model_get_iter(model, &iter, pkd->click_tpath))
734 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
735 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
743 if (edit_existing && !name) return;
746 cdd = g_new0(ConfDialogData, 1);
748 cdd->click_tpath = pkd->click_tpath;
749 pkd->click_tpath = NULL;
750 cdd->is_keyword = is_keyword;
751 cdd->edit_existing = edit_existing;
753 cdd->gd = gd = generic_dialog_new(name ? _("Edit keyword") : _("Add keywords"), "keyword_edit",
755 bar_pane_keywords_edit_cancel_cb, cdd);
756 g_signal_connect(G_OBJECT(gd->dialog), "destroy",
757 G_CALLBACK(bar_pane_keywords_edit_destroy_cb), cdd);
760 generic_dialog_add_message(gd, NULL, name ? _("Configure keyword") : _("Add keyword"), NULL);
762 generic_dialog_add_button(gd, GTK_STOCK_OK, NULL,
763 bar_pane_keywords_edit_ok_cb, TRUE);
765 table = pref_table_new(gd->vbox, 3, 1, FALSE, TRUE);
766 pref_table_label(table, 0, 0, _("Keyword:"), 1.0);
767 cdd->edit_widget = gtk_entry_new();
768 gtk_widget_set_size_request(cdd->edit_widget, 300, -1);
769 if (name) gtk_entry_set_text(GTK_ENTRY(cdd->edit_widget), name);
770 gtk_table_attach_defaults(GTK_TABLE(table), cdd->edit_widget, 1, 2, 0, 1);
771 /* here could eventually be a text view instead of entry */
772 generic_dialog_attach_default(gd, cdd->edit_widget);
773 gtk_widget_show(cdd->edit_widget);
775 group = pref_group_new(gd->vbox, FALSE, _("Keyword type:"), GTK_ORIENTATION_VERTICAL);
777 button = pref_radiobutton_new(group, NULL, _("Active keyword"),
779 G_CALLBACK(bar_pane_keywords_conf_set_kw), cdd);
780 button = pref_radiobutton_new(group, button, _("Helper"),
782 G_CALLBACK(bar_pane_keywords_conf_set_helper), cdd);
785 gtk_widget_show(gd->dialog);
792 *-------------------------------------------------------------------
794 *-------------------------------------------------------------------
797 static void bar_pane_keywords_edit_dialog_cb(GtkWidget *menu_widget, gpointer data)
799 PaneKeywordsData *pkd = data;
800 bar_pane_keywords_edit_dialog(pkd, TRUE);
803 static void bar_pane_keywords_add_dialog_cb(GtkWidget *menu_widget, gpointer data)
805 PaneKeywordsData *pkd = data;
806 bar_pane_keywords_edit_dialog(pkd, FALSE);
809 static void bar_pane_keywords_connect_mark_cb(GtkWidget *menu_widget, gpointer data)
811 PaneKeywordsData *pkd = data;
816 GtkTreeModel *keyword_tree;
819 gint mark = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menu_widget), "mark")) - 1;
821 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
822 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
824 if (!pkd->click_tpath) return;
825 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
827 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
829 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
831 meta_data_connect_mark_with_keyword(keyword_tree, &kw_iter, mark);
833 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
834 // bar_pane_keywords_update(pkd);
838 static void bar_pane_keywords_delete_cb(GtkWidget *menu_widget, gpointer data)
840 PaneKeywordsData *pkd = data;
844 GtkTreeModel *keyword_tree;
847 if (!pkd->click_tpath) return;
849 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
850 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
852 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
853 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
855 keyword_delete(GTK_TREE_STORE(keyword_tree), &kw_iter);
858 static void bar_pane_keywords_menu_popup(GtkWidget *widget, PaneKeywordsData *pkd, gint x, gint y)
861 GtkTreeViewDropPosition pos;
863 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
864 pkd->click_tpath = NULL;
865 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(pkd->keyword_treeview), x, y, &pkd->click_tpath, &pos);
867 menu = popup_menu_short_lived();
869 if (pkd->click_tpath)
878 GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
881 gtk_tree_model_get_iter(model, &iter, pkd->click_tpath);
884 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
885 FILTER_KEYWORD_COLUMN_MARK, &mark, -1);
887 text = g_strdup_printf(_("Edit \"%s\""), name);
888 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_edit_dialog_cb), pkd);
890 text = g_strdup_printf(_("Delete \"%s\""), name);
891 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_delete_cb), pkd);
894 submenu = gtk_menu_new();
895 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
897 text = g_strdup_printf(_("Mark %d"), i + 1);
898 item = menu_item_add(submenu, text, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
899 g_object_set_data(G_OBJECT(item), "mark", GINT_TO_POINTER(i + 1));
903 text = g_strdup_printf(_("Connect \"%s\" to mark"), name);
904 item = menu_item_add(menu, text, NULL, NULL);
905 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
910 text = g_strdup_printf(_("Disconnect \"%s\" from mark %s"), name, mark);
911 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
915 menu_item_add_divider(menu);
920 menu_item_add_stock(menu, _("Add keyword"), GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_add_dialog_cb), pkd);
922 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 0, GDK_CURRENT_TIME);
926 static gboolean bar_pane_keywords_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
928 PaneKeywordsData *pkd = data;
929 if (bevent->button == MOUSE_BUTTON_RIGHT)
931 bar_pane_keywords_menu_popup(widget, pkd, bevent->x, bevent->y);
938 *-------------------------------------------------------------------
940 *-------------------------------------------------------------------
943 void bar_pane_keywords_close(GtkWidget *bar)
945 PaneKeywordsData *pkd;
947 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
950 gtk_widget_destroy(pkd->widget);
953 static void bar_pane_keywords_destroy(GtkWidget *widget, gpointer data)
955 PaneKeywordsData *pkd = data;
957 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
959 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
961 file_data_unref(pkd->fd);
968 GtkWidget *bar_pane_keywords_new(const gchar *title, const gchar *key, gboolean expanded)
970 PaneKeywordsData *pkd;
973 GtkTextBuffer *buffer;
975 GtkTreeViewColumn *column;
976 GtkCellRenderer *renderer;
978 pkd = g_new0(PaneKeywordsData, 1);
980 pkd->pane.pane_set_fd = bar_pane_keywords_set_fd;
981 pkd->pane.pane_event = bar_pane_keywords_event;
982 pkd->pane.pane_write_config = bar_pane_keywords_write_config;
983 pkd->pane.title = bar_pane_expander_title(title);
985 pkd->pane.expanded = expanded;
987 pkd->key = g_strdup(key);
990 hbox = gtk_hbox_new(FALSE, PREF_PAD_GAP);
993 g_object_set_data(G_OBJECT(pkd->widget), "pane_data", pkd);
994 g_signal_connect(G_OBJECT(pkd->widget), "destroy",
995 G_CALLBACK(bar_pane_keywords_destroy), pkd);
996 gtk_widget_show(hbox);
998 scrolled = gtk_scrolled_window_new(NULL, NULL);
999 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1000 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1001 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1002 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1003 gtk_widget_show(scrolled);
1005 pkd->keyword_view = gtk_text_view_new();
1006 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_view);
1007 g_signal_connect(G_OBJECT(pkd->keyword_view), "populate-popup",
1008 G_CALLBACK(bar_pane_keywords_populate_popup_cb), pkd);
1009 gtk_widget_show(pkd->keyword_view);
1011 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
1012 g_signal_connect(G_OBJECT(buffer), "changed",
1013 G_CALLBACK(bar_pane_keywords_changed), pkd);
1015 scrolled = gtk_scrolled_window_new(NULL, NULL);
1016 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1017 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1018 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1019 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1020 gtk_widget_show(scrolled);
1023 if (!keyword_tree) keyword_tree_new_default();
1025 store = gtk_tree_model_filter_new(GTK_TREE_MODEL(keyword_tree), NULL);
1027 gtk_tree_model_filter_set_modify_func(GTK_TREE_MODEL_FILTER(store),
1028 FILTER_KEYWORD_COLUMN_COUNT,
1029 filter_keyword_column_types,
1030 bar_pane_keywords_filter_modify,
1034 pkd->keyword_treeview = gtk_tree_view_new_with_model(store);
1035 g_object_unref(store);
1037 gtk_widget_set_size_request(pkd->keyword_treeview, -1, 400);
1039 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(pkd->keyword_treeview), FALSE);
1041 // gtk_tree_view_set_search_column(GTK_TREE_VIEW(pkd->keyword_treeview), FILTER_KEYWORD_COLUMN_);
1043 column = gtk_tree_view_column_new();
1044 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
1046 renderer = gtk_cell_renderer_text_new();
1047 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1049 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_MARK);
1051 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1053 column = gtk_tree_view_column_new();
1054 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1055 renderer = gtk_cell_renderer_toggle_new();
1056 gtk_tree_view_column_pack_start(column, renderer, FALSE);
1057 gtk_tree_view_column_add_attribute(column, renderer, "active", FILTER_KEYWORD_COLUMN_TOGGLE);
1058 gtk_tree_view_column_add_attribute(column, renderer, "visible", FILTER_KEYWORD_COLUMN_IS_KEYWORD);
1059 g_signal_connect(G_OBJECT(renderer), "toggled",
1060 G_CALLBACK(bar_pane_keywords_keyword_toggle), pkd);
1062 renderer = gtk_cell_renderer_text_new();
1063 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1064 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_NAME);
1066 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1067 gtk_tree_view_set_expander_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1069 gtk_drag_source_set(pkd->keyword_treeview,
1070 GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
1071 bar_pane_keywords_drag_types, n_keywords_drag_types,
1072 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
1074 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_get",
1075 G_CALLBACK(bar_pane_keywords_dnd_get), pkd);
1077 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_begin",
1078 G_CALLBACK(bar_pane_keywords_dnd_begin), pkd);
1079 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_end",
1080 G_CALLBACK(bar_pane_keywords_dnd_end), pkd);
1082 gtk_drag_dest_set(pkd->keyword_treeview,
1083 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
1084 bar_pane_keywords_drop_types, n_keywords_drop_types,
1085 GDK_ACTION_COPY | GDK_ACTION_MOVE);
1087 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_received",
1088 G_CALLBACK(bar_pane_keywords_dnd_receive), pkd);
1090 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_motion",
1091 G_CALLBACK(bar_pane_keywords_dnd_motion), pkd);
1093 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "button_press_event",
1094 G_CALLBACK(bar_pane_keywords_menu_cb), pkd);
1096 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_treeview);
1097 gtk_widget_show(pkd->keyword_treeview);
1099 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
1104 GtkWidget *bar_pane_keywords_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
1106 gchar *title = g_strdup(_("NoName"));
1107 gchar *key = g_strdup(COMMENT_KEY);
1108 gboolean expanded = TRUE;
1110 while (*attribute_names)
1112 const gchar *option = *attribute_names++;
1113 const gchar *value = *attribute_values++;
1115 if (READ_CHAR_FULL("pane.title", title)) continue;
1116 if (READ_CHAR_FULL("key", key)) continue;
1117 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
1120 DEBUG_1("unknown attribute %s = %s", option, value);
1123 return bar_pane_keywords_new(title, key, expanded);
1126 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */