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;
118 typedef struct _ConfDialogData ConfDialogData;
119 struct _ConfDialogData
121 PaneKeywordsData *pkd;
122 GtkTreePath *click_tpath;
126 GtkWidget *edit_widget;
129 gboolean edit_existing;
132 static GList *bar_list = NULL;
135 static void bar_pane_keywords_write(PaneKeywordsData *pkd)
139 if (!pkd->fd) return;
141 list = keyword_list_pull(pkd->keyword_view);
143 metadata_write_list(pkd->fd, KEYWORD_KEY, list);
145 string_list_free(list);
148 gboolean bar_keyword_tree_expand_if_set_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
150 PaneKeywordsData *pkd = data;
153 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set, -1);
155 if (set && !gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
157 gtk_tree_view_expand_to_path(GTK_TREE_VIEW(pkd->keyword_treeview), path);
162 gboolean bar_keyword_tree_collapse_if_unset_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
164 PaneKeywordsData *pkd = data;
168 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set,
169 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
171 if (is_keyword && !set && gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
173 gtk_tree_view_collapse_row(GTK_TREE_VIEW(pkd->keyword_treeview), path);
178 static void bar_keyword_tree_sync(PaneKeywordsData *pkd)
182 GtkTreeModel *keyword_tree;
185 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
186 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
188 keywords = keyword_list_pull(pkd->keyword_view);
189 keyword_show_set_in(GTK_TREE_STORE(keyword_tree), model, keywords);
190 if (pkd->hide_unchecked) keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
191 string_list_free(keywords);
193 gtk_tree_model_filter_refilter(GTK_TREE_MODEL_FILTER(model));
195 if (pkd->expand_checked) gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
196 if (pkd->collapse_unchecked) gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
199 static void bar_pane_keywords_keyword_update_all(void)
206 PaneKeywordsData *pkd;
212 bar_keyword_tree_sync(pkd);
216 static void bar_pane_keywords_update(PaneKeywordsData *pkd)
218 GList *keywords = NULL;
219 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
221 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
223 keywords = metadata_read_list(pkd->fd, KEYWORD_KEY, METADATA_PLAIN);
224 keyword_list_push(pkd->keyword_view, keywords);
225 bar_keyword_tree_sync(pkd);
226 string_list_free(keywords);
228 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
232 void bar_pane_keywords_set_fd(GtkWidget *pane, FileData *fd)
234 PaneKeywordsData *pkd;
236 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
239 file_data_unref(pkd->fd);
240 pkd->fd = file_data_ref(fd);
242 bar_pane_keywords_update(pkd);
245 static void bar_pane_keywords_write_config(GtkWidget *pane, GString *outstr, gint indent)
247 PaneKeywordsData *pkd;
249 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
252 WRITE_STRING("<pane_keywords\n");
254 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(pkd->pane.title)));
255 WRITE_BOOL(*pkd, pane.expanded);
256 WRITE_CHAR(*pkd, key);
258 WRITE_STRING("/>\n");
261 gint bar_pane_keywords_event(GtkWidget *bar, GdkEvent *event)
263 PaneKeywordsData *pkd;
265 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
266 if (!pkd) return FALSE;
268 if (GTK_WIDGET_HAS_FOCUS(pkd->keyword_view)) return gtk_widget_event(pkd->keyword_view, event);
273 static void bar_pane_keywords_keyword_toggle(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
275 PaneKeywordsData *pkd = data;
281 GtkTreeIter child_iter;
282 GtkTreeModel *keyword_tree;
284 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
286 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
288 tpath = gtk_tree_path_new_from_string(path);
289 gtk_tree_model_get_iter(model, &iter, tpath);
290 gtk_tree_path_free(tpath);
292 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_TOGGLE, &active, -1);
296 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
297 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
299 list = keyword_list_pull(pkd->keyword_view);
301 keyword_tree_set(keyword_tree, &child_iter, &list);
303 keyword_tree_reset(keyword_tree, &child_iter, &list);
305 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
306 keyword_list_push(pkd->keyword_view, list);
307 string_list_free(list);
308 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
310 /* call this just once in the end */
311 bar_pane_keywords_changed(keyword_buffer, pkd);
313 bar_pane_keywords_change calls bar_keyword_tree_sync, no need to do it again
314 bar_keyword_tree_sync(pkd);
318 void bar_pane_keywords_filter_modify(GtkTreeModel *model, GtkTreeIter *iter, GValue *value, gint column, gpointer data)
320 PaneKeywordsData *pkd = data;
321 GtkTreeModel *keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
322 GtkTreeIter child_iter;
324 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, iter);
326 memset(value, 0, sizeof (GValue));
330 case FILTER_KEYWORD_COLUMN_TOGGLE:
332 GList *keywords = keyword_list_pull(pkd->keyword_view);
333 gboolean set = keyword_tree_is_set(keyword_tree, &child_iter, keywords);
334 string_list_free(keywords);
336 g_value_init(value, G_TYPE_BOOLEAN);
337 g_value_set_boolean(value, set);
340 case FILTER_KEYWORD_COLUMN_MARK:
341 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_MARK, value);
343 case FILTER_KEYWORD_COLUMN_NAME:
344 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_NAME, value);
346 case FILTER_KEYWORD_COLUMN_IS_KEYWORD:
347 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_IS_KEYWORD, value);
354 gboolean bar_pane_keywords_filter_visible(GtkTreeModel *keyword_tree, GtkTreeIter *iter, gpointer data)
356 GtkTreeModel *filter = data;
358 return !keyword_is_hidden_in(keyword_tree, iter, filter);
361 static void bar_pane_keywords_set_selection(PaneKeywordsData *pkd, gboolean append)
363 GList *keywords = NULL;
367 keywords = keyword_list_pull(pkd->keyword_view);
369 list = layout_selection_list(pkd->pane.lw);
373 FileData *fd = work->data;
378 metadata_append_list(fd, KEYWORD_KEY, keywords);
382 metadata_write_list(fd, KEYWORD_KEY, keywords);
387 string_list_free(keywords);
390 static void bar_pane_keywords_sel_add_cb(GtkWidget *button, gpointer data)
392 PaneKeywordsData *pkd = data;
394 bar_pane_keywords_set_selection(pkd, TRUE);
397 static void bar_pane_keywords_sel_replace_cb(GtkWidget *button, gpointer data)
399 PaneKeywordsData *pkd = data;
401 bar_pane_keywords_set_selection(pkd, FALSE);
404 static void bar_pane_keywords_populate_popup_cb(GtkTextView *textview, GtkMenu *menu, gpointer data)
406 PaneKeywordsData *pkd = data;
408 menu_item_add_divider(GTK_WIDGET(menu));
409 menu_item_add_stock(GTK_WIDGET(menu), _("Add keywords to selected files"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_keywords_sel_add_cb), pkd);
410 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);
414 static void bar_pane_keywords_notify_cb(FileData *fd, NotifyType type, gpointer data)
416 PaneKeywordsData *pkd = data;
417 if ((type & (NOTIFY_REREAD | NOTIFY_CHANGE | NOTIFY_METADATA)) && fd == pkd->fd) bar_pane_keywords_update(pkd);
420 static gboolean bar_pane_keywords_changed_idle_cb(gpointer data)
422 PaneKeywordsData *pkd = data;
424 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
425 bar_pane_keywords_write(pkd);
426 bar_keyword_tree_sync(pkd);
427 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
432 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data)
434 PaneKeywordsData *pkd = data;
436 if (pkd->idle_id != -1) return;
437 /* higher prio than redraw */
438 pkd->idle_id = g_idle_add_full(G_PRIORITY_HIGH_IDLE, bar_pane_keywords_changed_idle_cb, pkd, NULL);
443 *-------------------------------------------------------------------
445 *-------------------------------------------------------------------
449 static GtkTargetEntry bar_pane_keywords_drag_types[] = {
450 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
451 { "text/plain", 0, TARGET_TEXT_PLAIN }
453 static gint n_keywords_drag_types = 2;
456 static GtkTargetEntry bar_pane_keywords_drop_types[] = {
457 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
458 { "text/plain", 0, TARGET_TEXT_PLAIN }
460 static gint n_keywords_drop_types = 2;
463 static void bar_pane_keywords_dnd_get(GtkWidget *tree_view, GdkDragContext *context,
464 GtkSelectionData *selection_data, guint info,
465 guint time, gpointer data)
469 GtkTreeIter child_iter;
470 GtkTreeModel *keyword_tree;
472 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
474 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
476 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
477 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
481 case TARGET_APP_KEYWORD_PATH:
483 GList *path = keyword_tree_get_path(keyword_tree, &child_iter);
484 gtk_selection_data_set(selection_data, selection_data->target,
485 8, (gpointer) &path, sizeof(path));
489 case TARGET_TEXT_PLAIN:
492 gchar *name = keyword_get_name(keyword_tree, &child_iter);
493 gtk_selection_data_set_text(selection_data, name, -1);
500 static void bar_pane_keywords_dnd_begin(GtkWidget *tree_view, GdkDragContext *context, gpointer data)
504 GtkTreeIter child_iter;
505 GtkTreeModel *keyword_tree;
508 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
510 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
512 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
513 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
515 name = keyword_get_name(keyword_tree, &child_iter);
517 dnd_set_drag_label(tree_view, context, name);
522 static void bar_pane_keywords_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
527 static gboolean bar_pane_keywords_dnd_can_move(GtkTreeModel *keyword_tree, GtkTreeIter *src_kw_iter, GtkTreeIter *dest_kw_iter)
532 if (dest_kw_iter && keyword_same_parent(keyword_tree, src_kw_iter, dest_kw_iter))
534 return TRUE; /* reordering of siblings is ok */
536 if (!dest_kw_iter && !gtk_tree_model_iter_parent(keyword_tree, &parent, src_kw_iter))
538 return TRUE; /* reordering of top-level siblings is ok */
541 src_name = keyword_get_name(keyword_tree, src_kw_iter);
542 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, src_name, FALSE))
551 static gboolean bar_pane_keywords_dnd_skip_existing(GtkTreeModel *keyword_tree, GtkTreeIter *dest_kw_iter, GList **keywords)
553 /* we have to find at least one keyword that does not already exist as a sibling of dest_kw_iter */
554 GList *work = *keywords;
557 gchar *keyword = work->data;
558 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, keyword, FALSE))
560 GList *next = work->next;
562 *keywords = g_list_delete_link(*keywords, work);
573 static void bar_pane_keywords_dnd_receive(GtkWidget *tree_view, GdkDragContext *context,
575 GtkSelectionData *selection_data, guint info,
576 guint time, gpointer data)
578 PaneKeywordsData *pkd = data;
579 GtkTreePath *tpath = NULL;
580 GtkTreeViewDropPosition pos;
583 GtkTreeModel *keyword_tree;
584 gboolean src_valid = FALSE;
585 GList *new_keywords = NULL;
588 /* iterators for keyword_tree */
589 GtkTreeIter src_kw_iter;
590 GtkTreeIter dest_kw_iter;
591 GtkTreeIter new_kw_iter;
593 g_signal_stop_emission_by_name(tree_view, "drag_data_received");
595 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
596 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
598 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
599 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), NULL, pos);
603 case TARGET_APP_KEYWORD_PATH:
605 GList *path = *(gpointer *)selection_data->data;
606 src_valid = keyword_tree_get_iter(keyword_tree, &src_kw_iter, path);
607 string_list_free(path);
611 new_keywords = string_to_keywords_list((gchar *)selection_data->data);
617 GtkTreeIter dest_iter;
618 gtk_tree_model_get_iter(model, &dest_iter, tpath);
619 gtk_tree_path_free(tpath);
620 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &dest_kw_iter, &dest_iter);
622 if (src_valid && gtk_tree_store_is_ancestor(GTK_TREE_STORE(keyword_tree), &src_kw_iter, &dest_kw_iter))
624 /* can't move to it's own child */
628 if (src_valid && keyword_compare(keyword_tree, &src_kw_iter, &dest_kw_iter) == 0)
630 /* can't move to itself */
634 if ((pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE || pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER) &&
635 !gtk_tree_model_iter_has_child(keyword_tree, &dest_kw_iter))
637 /* the node has no children, all keywords can be added */
638 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &dest_kw_iter);
642 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, &dest_kw_iter))
644 /* the keyword can't be moved if the same name already exist */
647 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, &dest_kw_iter, &new_keywords))
649 /* the keywords can't be added if the same name already exist */
655 case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
656 case GTK_TREE_VIEW_DROP_BEFORE:
657 gtk_tree_store_insert_before(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
659 case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
660 case GTK_TREE_VIEW_DROP_AFTER:
661 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
669 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, NULL))
671 /* the keyword can't be moved if the same name already exist */
674 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, NULL, &new_keywords))
676 /* the keywords can't be added if the same name already exist */
679 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL);
685 keyword_move_recursive(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &src_kw_iter);
691 gchar *keyword = work->data;
692 keyword_set(GTK_TREE_STORE(keyword_tree), &new_kw_iter, keyword, TRUE);
698 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &new_kw_iter);
702 string_list_free(new_keywords);
703 bar_keyword_tree_sync(pkd);
706 static gint bar_pane_keywords_dnd_motion(GtkWidget *tree_view, GdkDragContext *context,
707 gint x, gint y, guint time, gpointer data)
709 GtkTreePath *tpath = NULL;
710 GtkTreeViewDropPosition pos;
711 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
715 GtkTreeIter dest_iter;
716 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
717 gtk_tree_model_get_iter(model, &dest_iter, tpath);
718 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE && gtk_tree_model_iter_has_child(model, &dest_iter))
719 pos = GTK_TREE_VIEW_DROP_BEFORE;
721 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER && gtk_tree_model_iter_has_child(model, &dest_iter))
722 pos = GTK_TREE_VIEW_DROP_AFTER;
725 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), tpath, pos);
726 gtk_tree_path_free(tpath);
728 if (tree_view == gtk_drag_get_source_widget(context))
729 gdk_drag_status(context, GDK_ACTION_MOVE, time);
731 gdk_drag_status(context, GDK_ACTION_COPY, time);
737 *-------------------------------------------------------------------
739 *-------------------------------------------------------------------
742 static void bar_pane_keywords_edit_destroy_cb(GtkWidget *widget, gpointer data)
744 ConfDialogData *cdd = data;
745 gtk_tree_path_free(cdd->click_tpath);
750 static void bar_pane_keywords_edit_cancel_cb(GenericDialog *gd, gpointer data)
755 static void bar_pane_keywords_edit_ok_cb(GenericDialog *gd, gpointer data)
757 ConfDialogData *cdd = data;
758 PaneKeywordsData *pkd = cdd->pkd;
761 GtkTreeModel *keyword_tree;
764 gboolean have_dest = FALSE;
768 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
769 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
771 if (cdd->click_tpath)
774 if (gtk_tree_model_get_iter(model, &iter, cdd->click_tpath))
776 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
781 if (cdd->edit_existing && !have_dest) return;
783 keywords = keyword_list_pull(cdd->edit_widget);
785 if (cdd->edit_existing)
787 if (keywords && keywords->data && /* there should be one keyword */
788 !keyword_exists(keyword_tree, NULL, &kw_iter, keywords->data, TRUE))
790 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, keywords->data, cdd->is_keyword);
795 GList *work = keywords;
800 if (keyword_exists(keyword_tree, NULL, have_dest ? &kw_iter : NULL, work->data, FALSE))
807 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &kw_iter);
811 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &add, NULL);
815 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, work->data, cdd->is_keyword);
819 string_list_free(keywords);
822 static void bar_pane_keywords_conf_set_helper(GtkWidget *widget, gpointer data)
824 ConfDialogData *cdd = data;
825 cdd->is_keyword = FALSE;
828 static void bar_pane_keywords_conf_set_kw(GtkWidget *widget, gpointer data)
830 ConfDialogData *cdd = data;
831 cdd->is_keyword = TRUE;
836 static void bar_pane_keywords_edit_dialog(PaneKeywordsData *pkd, gboolean edit_existing)
845 gboolean is_keyword = TRUE;
848 if (edit_existing && pkd->click_tpath)
852 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
854 if (gtk_tree_model_get_iter(model, &iter, pkd->click_tpath))
856 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
857 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
865 if (edit_existing && !name) return;
868 cdd = g_new0(ConfDialogData, 1);
870 cdd->click_tpath = pkd->click_tpath;
871 pkd->click_tpath = NULL;
872 cdd->is_keyword = is_keyword;
873 cdd->edit_existing = edit_existing;
875 cdd->gd = gd = generic_dialog_new(name ? _("Edit keyword") : _("Add keywords"), "keyword_edit",
877 bar_pane_keywords_edit_cancel_cb, cdd);
878 g_signal_connect(G_OBJECT(gd->dialog), "destroy",
879 G_CALLBACK(bar_pane_keywords_edit_destroy_cb), cdd);
882 generic_dialog_add_message(gd, NULL, name ? _("Configure keyword") : _("Add keyword"), NULL);
884 generic_dialog_add_button(gd, GTK_STOCK_OK, NULL,
885 bar_pane_keywords_edit_ok_cb, TRUE);
887 table = pref_table_new(gd->vbox, 3, 1, FALSE, TRUE);
888 pref_table_label(table, 0, 0, _("Keyword:"), 1.0);
889 cdd->edit_widget = gtk_entry_new();
890 gtk_widget_set_size_request(cdd->edit_widget, 300, -1);
891 if (name) gtk_entry_set_text(GTK_ENTRY(cdd->edit_widget), name);
892 gtk_table_attach_defaults(GTK_TABLE(table), cdd->edit_widget, 1, 2, 0, 1);
893 /* here could eventually be a text view instead of entry */
894 generic_dialog_attach_default(gd, cdd->edit_widget);
895 gtk_widget_show(cdd->edit_widget);
897 group = pref_group_new(gd->vbox, FALSE, _("Keyword type:"), GTK_ORIENTATION_VERTICAL);
899 button = pref_radiobutton_new(group, NULL, _("Active keyword"),
901 G_CALLBACK(bar_pane_keywords_conf_set_kw), cdd);
902 button = pref_radiobutton_new(group, button, _("Helper"),
904 G_CALLBACK(bar_pane_keywords_conf_set_helper), cdd);
907 gtk_widget_show(gd->dialog);
914 *-------------------------------------------------------------------
916 *-------------------------------------------------------------------
919 static void bar_pane_keywords_edit_dialog_cb(GtkWidget *menu_widget, gpointer data)
921 PaneKeywordsData *pkd = data;
922 bar_pane_keywords_edit_dialog(pkd, TRUE);
925 static void bar_pane_keywords_add_dialog_cb(GtkWidget *menu_widget, gpointer data)
927 PaneKeywordsData *pkd = data;
928 bar_pane_keywords_edit_dialog(pkd, FALSE);
931 static void bar_pane_keywords_connect_mark_cb(GtkWidget *menu_widget, gpointer data)
933 PaneKeywordsData *pkd = data;
938 GtkTreeModel *keyword_tree;
941 gint mark = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menu_widget), "mark")) - 1;
943 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
944 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
946 if (!pkd->click_tpath) return;
947 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
949 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
951 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
953 meta_data_connect_mark_with_keyword(keyword_tree, &kw_iter, mark);
955 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
956 // bar_pane_keywords_update(pkd);
960 static void bar_pane_keywords_delete_cb(GtkWidget *menu_widget, gpointer data)
962 PaneKeywordsData *pkd = data;
966 GtkTreeModel *keyword_tree;
969 if (!pkd->click_tpath) return;
971 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
972 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
974 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
975 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
977 keyword_delete(GTK_TREE_STORE(keyword_tree), &kw_iter);
980 static void bar_pane_keywords_hide_cb(GtkWidget *menu_widget, gpointer data)
982 PaneKeywordsData *pkd = data;
986 GtkTreeModel *keyword_tree;
989 if (!pkd->click_tpath) return;
991 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
992 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
994 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
995 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
997 keyword_hide_in(GTK_TREE_STORE(keyword_tree), &kw_iter, model);
1000 static void bar_pane_keywords_show_all_cb(GtkWidget *menu_widget, gpointer data)
1002 PaneKeywordsData *pkd = data;
1003 GtkTreeModel *model;
1005 GtkTreeModel *keyword_tree;
1007 pkd->hide_unchecked = FALSE;
1009 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1010 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1012 keyword_show_all_in(GTK_TREE_STORE(keyword_tree), model);
1014 if (!pkd->collapse_unchecked) gtk_tree_view_expand_all(GTK_TREE_VIEW(pkd->keyword_treeview));
1015 bar_keyword_tree_sync(pkd);
1018 static void bar_pane_keywords_expand_checked_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_expand_if_set_cb, pkd);
1027 static void bar_pane_keywords_collapse_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1029 PaneKeywordsData *pkd = data;
1030 GtkTreeModel *model;
1032 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1033 gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
1036 static void bar_pane_keywords_hide_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1038 PaneKeywordsData *pkd = data;
1039 GtkTreeModel *model;
1041 GtkTreeModel *keyword_tree;
1044 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1045 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1047 keywords = keyword_list_pull(pkd->keyword_view);
1048 keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
1049 string_list_free(keywords);
1050 bar_keyword_tree_sync(pkd);
1053 static void bar_pane_keywords_expand_checked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1055 PaneKeywordsData *pkd = data;
1056 pkd->expand_checked = !pkd->expand_checked;
1057 bar_keyword_tree_sync(pkd);
1060 static void bar_pane_keywords_collapse_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1062 PaneKeywordsData *pkd = data;
1063 pkd->collapse_unchecked = !pkd->collapse_unchecked;
1064 bar_keyword_tree_sync(pkd);
1067 static void bar_pane_keywords_hide_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1069 PaneKeywordsData *pkd = data;
1070 pkd->hide_unchecked = !pkd->hide_unchecked;
1071 bar_keyword_tree_sync(pkd);
1074 static void bar_pane_keywords_menu_popup(GtkWidget *widget, PaneKeywordsData *pkd, gint x, gint y)
1079 GtkTreeViewDropPosition pos;
1081 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1082 pkd->click_tpath = NULL;
1083 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(pkd->keyword_treeview), x, y, &pkd->click_tpath, &pos);
1085 menu = popup_menu_short_lived();
1087 if (pkd->click_tpath)
1094 GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1097 gtk_tree_model_get_iter(model, &iter, pkd->click_tpath);
1100 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
1101 FILTER_KEYWORD_COLUMN_MARK, &mark, -1);
1103 text = g_strdup_printf(_("Hide \"%s\""), name);
1104 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_hide_cb), pkd);
1107 submenu = gtk_menu_new();
1108 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
1110 text = g_strdup_printf(_("Mark %d"), i + 1);
1111 item = menu_item_add(submenu, text, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1112 g_object_set_data(G_OBJECT(item), "mark", GINT_TO_POINTER(i + 1));
1115 text = g_strdup_printf(_("Connect \"%s\" to mark"), name);
1116 item = menu_item_add(menu, text, NULL, NULL);
1117 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1120 menu_item_add_divider(menu);
1122 text = g_strdup_printf(_("Edit \"%s\""), name);
1123 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_edit_dialog_cb), pkd);
1125 text = g_strdup_printf(_("Delete \"%s\""), name);
1126 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_delete_cb), pkd);
1130 if (mark && mark[0])
1132 text = g_strdup_printf(_("Disconnect \"%s\" from mark %s"), name, mark);
1133 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1137 menu_item_add_divider(menu);
1144 menu_item_add(menu, _("Expand checked"), G_CALLBACK(bar_pane_keywords_expand_checked_cb), pkd);
1145 menu_item_add(menu, _("Collapse unchecked"), G_CALLBACK(bar_pane_keywords_collapse_unchecked_cb), pkd);
1146 menu_item_add(menu, _("Hide unchecked"), G_CALLBACK(bar_pane_keywords_hide_unchecked_cb), pkd);
1147 menu_item_add(menu, _("Show all"), G_CALLBACK(bar_pane_keywords_show_all_cb), pkd);
1149 submenu = gtk_menu_new();
1150 item = menu_item_add(menu, _("On any change"), NULL, NULL);
1151 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1153 menu_item_add_check(submenu, _("Expand checked"), pkd->expand_checked, G_CALLBACK(bar_pane_keywords_expand_checked_toggle_cb), pkd);
1154 menu_item_add_check(submenu, _("Collapse unchecked"), pkd->collapse_unchecked, G_CALLBACK(bar_pane_keywords_collapse_unchecked_toggle_cb), pkd);
1155 menu_item_add_check(submenu, _("Hide unchecked"), pkd->hide_unchecked, G_CALLBACK(bar_pane_keywords_hide_unchecked_toggle_cb), pkd);
1157 menu_item_add_divider(menu);
1159 menu_item_add_stock(menu, _("Add keyword"), GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_add_dialog_cb), pkd);
1161 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 0, GDK_CURRENT_TIME);
1165 static gboolean bar_pane_keywords_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
1167 PaneKeywordsData *pkd = data;
1168 if (bevent->button == MOUSE_BUTTON_RIGHT)
1170 bar_pane_keywords_menu_popup(widget, pkd, bevent->x, bevent->y);
1177 *-------------------------------------------------------------------
1179 *-------------------------------------------------------------------
1182 void bar_pane_keywords_close(GtkWidget *bar)
1184 PaneKeywordsData *pkd;
1186 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
1189 gtk_widget_destroy(pkd->widget);
1192 static void bar_pane_keywords_destroy(GtkWidget *widget, gpointer data)
1194 PaneKeywordsData *pkd = data;
1196 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1197 if (pkd->idle_id != -1) g_source_remove(pkd->idle_id);
1198 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
1200 file_data_unref(pkd->fd);
1207 GtkWidget *bar_pane_keywords_new(const gchar *title, const gchar *key, gboolean expanded)
1209 PaneKeywordsData *pkd;
1211 GtkWidget *scrolled;
1212 GtkTextBuffer *buffer;
1213 GtkTreeModel *store;
1214 GtkTreeViewColumn *column;
1215 GtkCellRenderer *renderer;
1217 pkd = g_new0(PaneKeywordsData, 1);
1219 pkd->pane.pane_set_fd = bar_pane_keywords_set_fd;
1220 pkd->pane.pane_event = bar_pane_keywords_event;
1221 pkd->pane.pane_write_config = bar_pane_keywords_write_config;
1222 pkd->pane.title = bar_pane_expander_title(title);
1224 pkd->pane.expanded = expanded;
1226 pkd->key = g_strdup(key);
1228 pkd->expand_checked = TRUE;
1232 hbox = gtk_hbox_new(FALSE, PREF_PAD_GAP);
1235 g_object_set_data(G_OBJECT(pkd->widget), "pane_data", pkd);
1236 g_signal_connect(G_OBJECT(pkd->widget), "destroy",
1237 G_CALLBACK(bar_pane_keywords_destroy), pkd);
1238 gtk_widget_show(hbox);
1240 scrolled = gtk_scrolled_window_new(NULL, NULL);
1241 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1242 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1243 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1244 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1245 gtk_widget_show(scrolled);
1247 pkd->keyword_view = gtk_text_view_new();
1248 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_view);
1249 g_signal_connect(G_OBJECT(pkd->keyword_view), "populate-popup",
1250 G_CALLBACK(bar_pane_keywords_populate_popup_cb), pkd);
1251 gtk_widget_show(pkd->keyword_view);
1253 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
1254 g_signal_connect(G_OBJECT(buffer), "changed",
1255 G_CALLBACK(bar_pane_keywords_changed), pkd);
1257 scrolled = gtk_scrolled_window_new(NULL, NULL);
1258 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1259 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1260 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1261 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1262 gtk_widget_show(scrolled);
1265 if (!keyword_tree) keyword_tree_new_default();
1267 store = gtk_tree_model_filter_new(GTK_TREE_MODEL(keyword_tree), NULL);
1269 gtk_tree_model_filter_set_modify_func(GTK_TREE_MODEL_FILTER(store),
1270 FILTER_KEYWORD_COLUMN_COUNT,
1271 filter_keyword_column_types,
1272 bar_pane_keywords_filter_modify,
1275 gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(store),
1276 bar_pane_keywords_filter_visible,
1280 pkd->keyword_treeview = gtk_tree_view_new_with_model(store);
1281 g_object_unref(store);
1283 gtk_widget_set_size_request(pkd->keyword_treeview, -1, 400);
1285 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(pkd->keyword_treeview), FALSE);
1287 // gtk_tree_view_set_search_column(GTK_TREE_VIEW(pkd->keyword_treeview), FILTER_KEYWORD_COLUMN_);
1289 column = gtk_tree_view_column_new();
1290 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
1292 renderer = gtk_cell_renderer_text_new();
1293 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1295 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_MARK);
1297 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1299 column = gtk_tree_view_column_new();
1300 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1301 renderer = gtk_cell_renderer_toggle_new();
1302 gtk_tree_view_column_pack_start(column, renderer, FALSE);
1303 gtk_tree_view_column_add_attribute(column, renderer, "active", FILTER_KEYWORD_COLUMN_TOGGLE);
1304 gtk_tree_view_column_add_attribute(column, renderer, "visible", FILTER_KEYWORD_COLUMN_IS_KEYWORD);
1305 g_signal_connect(G_OBJECT(renderer), "toggled",
1306 G_CALLBACK(bar_pane_keywords_keyword_toggle), pkd);
1308 renderer = gtk_cell_renderer_text_new();
1309 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1310 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_NAME);
1312 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1313 gtk_tree_view_set_expander_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1315 gtk_drag_source_set(pkd->keyword_treeview,
1316 GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
1317 bar_pane_keywords_drag_types, n_keywords_drag_types,
1318 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
1320 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_get",
1321 G_CALLBACK(bar_pane_keywords_dnd_get), pkd);
1323 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_begin",
1324 G_CALLBACK(bar_pane_keywords_dnd_begin), pkd);
1325 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_end",
1326 G_CALLBACK(bar_pane_keywords_dnd_end), pkd);
1328 gtk_drag_dest_set(pkd->keyword_treeview,
1329 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
1330 bar_pane_keywords_drop_types, n_keywords_drop_types,
1331 GDK_ACTION_COPY | GDK_ACTION_MOVE);
1333 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_received",
1334 G_CALLBACK(bar_pane_keywords_dnd_receive), pkd);
1336 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_motion",
1337 G_CALLBACK(bar_pane_keywords_dnd_motion), pkd);
1339 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "button_press_event",
1340 G_CALLBACK(bar_pane_keywords_menu_cb), pkd);
1342 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_treeview);
1343 gtk_widget_show(pkd->keyword_treeview);
1345 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
1350 GtkWidget *bar_pane_keywords_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
1352 gchar *title = g_strdup(_("NoName"));
1353 gchar *key = g_strdup(COMMENT_KEY);
1354 gboolean expanded = TRUE;
1356 while (*attribute_names)
1358 const gchar *option = *attribute_names++;
1359 const gchar *value = *attribute_values++;
1361 if (READ_CHAR_FULL("pane.title", title)) continue;
1362 if (READ_CHAR_FULL("key", key)) continue;
1363 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
1366 DEBUG_1("unknown attribute %s = %s", option, value);
1369 return bar_pane_keywords_new(title, key, expanded);
1372 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */