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;
167 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set, -1);
169 if (!set && gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
171 gtk_tree_view_collapse_row(GTK_TREE_VIEW(pkd->keyword_treeview), path);
176 static void bar_keyword_tree_sync(PaneKeywordsData *pkd)
180 GtkTreeModel *keyword_tree;
183 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
184 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
186 keywords = keyword_list_pull(pkd->keyword_view);
187 keyword_show_set_in(GTK_TREE_STORE(keyword_tree), model, keywords);
188 if (pkd->hide_unchecked) keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
189 string_list_free(keywords);
191 gtk_tree_model_filter_refilter(GTK_TREE_MODEL_FILTER(model));
193 if (pkd->expand_checked) gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
194 if (pkd->collapse_unchecked) gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
197 static void bar_pane_keywords_keyword_update_all(void)
204 PaneKeywordsData *pkd;
210 bar_keyword_tree_sync(pkd);
214 static void bar_pane_keywords_update(PaneKeywordsData *pkd)
216 GList *keywords = NULL;
217 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
219 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
221 keywords = metadata_read_list(pkd->fd, KEYWORD_KEY, METADATA_PLAIN);
222 keyword_list_push(pkd->keyword_view, keywords);
223 bar_keyword_tree_sync(pkd);
224 string_list_free(keywords);
226 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
230 void bar_pane_keywords_set_fd(GtkWidget *pane, FileData *fd)
232 PaneKeywordsData *pkd;
234 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
237 file_data_unref(pkd->fd);
238 pkd->fd = file_data_ref(fd);
240 bar_pane_keywords_update(pkd);
243 static void bar_pane_keywords_write_config(GtkWidget *pane, GString *outstr, gint indent)
245 PaneKeywordsData *pkd;
247 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
250 WRITE_NL(); WRITE_STRING("<pane_keywords ");
251 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(pkd->pane.title)));
252 WRITE_BOOL(*pkd, pane.expanded);
253 WRITE_CHAR(*pkd, key);
257 gint bar_pane_keywords_event(GtkWidget *bar, GdkEvent *event)
259 PaneKeywordsData *pkd;
261 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
262 if (!pkd) return FALSE;
264 if (GTK_WIDGET_HAS_FOCUS(pkd->keyword_view)) return gtk_widget_event(pkd->keyword_view, event);
269 static void bar_pane_keywords_keyword_toggle(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
271 PaneKeywordsData *pkd = data;
277 GtkTreeIter child_iter;
278 GtkTreeModel *keyword_tree;
280 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
282 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
284 tpath = gtk_tree_path_new_from_string(path);
285 gtk_tree_model_get_iter(model, &iter, tpath);
286 gtk_tree_path_free(tpath);
288 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_TOGGLE, &active, -1);
292 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
293 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
295 list = keyword_list_pull(pkd->keyword_view);
297 keyword_tree_set(keyword_tree, &child_iter, &list);
299 keyword_tree_reset(keyword_tree, &child_iter, &list);
301 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
302 keyword_list_push(pkd->keyword_view, list);
303 string_list_free(list);
304 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
306 /* call this just once in the end */
307 bar_pane_keywords_changed(keyword_buffer, pkd);
309 bar_pane_keywords_change calls bar_keyword_tree_sync, no need to do it again
310 bar_keyword_tree_sync(pkd);
314 void bar_pane_keywords_filter_modify(GtkTreeModel *model, GtkTreeIter *iter, GValue *value, gint column, gpointer data)
316 PaneKeywordsData *pkd = data;
317 GtkTreeModel *keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
318 GtkTreeIter child_iter;
320 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, iter);
322 memset(value, 0, sizeof (GValue));
326 case FILTER_KEYWORD_COLUMN_TOGGLE:
328 GList *keywords = keyword_list_pull(pkd->keyword_view);
329 gboolean set = keyword_tree_is_set(keyword_tree, &child_iter, keywords);
330 string_list_free(keywords);
332 g_value_init(value, G_TYPE_BOOLEAN);
333 g_value_set_boolean(value, set);
336 case FILTER_KEYWORD_COLUMN_MARK:
337 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_MARK, value);
339 case FILTER_KEYWORD_COLUMN_NAME:
340 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_NAME, value);
342 case FILTER_KEYWORD_COLUMN_IS_KEYWORD:
343 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_IS_KEYWORD, value);
350 gboolean bar_pane_keywords_filter_visible(GtkTreeModel *keyword_tree, GtkTreeIter *iter, gpointer data)
352 GtkTreeModel *filter = data;
354 return !keyword_is_hidden_in(keyword_tree, iter, filter);
357 static void bar_pane_keywords_set_selection(PaneKeywordsData *pkd, gboolean append)
359 GList *keywords = NULL;
363 keywords = keyword_list_pull(pkd->keyword_view);
365 list = layout_selection_list(pkd->pane.lw);
369 FileData *fd = work->data;
374 metadata_append_list(fd, KEYWORD_KEY, keywords);
378 metadata_write_list(fd, KEYWORD_KEY, keywords);
383 string_list_free(keywords);
386 static void bar_pane_keywords_sel_add_cb(GtkWidget *button, gpointer data)
388 PaneKeywordsData *pkd = data;
390 bar_pane_keywords_set_selection(pkd, TRUE);
393 static void bar_pane_keywords_sel_replace_cb(GtkWidget *button, gpointer data)
395 PaneKeywordsData *pkd = data;
397 bar_pane_keywords_set_selection(pkd, FALSE);
400 static void bar_pane_keywords_populate_popup_cb(GtkTextView *textview, GtkMenu *menu, gpointer data)
402 PaneKeywordsData *pkd = data;
404 menu_item_add_divider(GTK_WIDGET(menu));
405 menu_item_add_stock(GTK_WIDGET(menu), _("Add keywords to selected files"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_keywords_sel_add_cb), pkd);
406 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);
410 static void bar_pane_keywords_notify_cb(FileData *fd, NotifyType type, gpointer data)
412 PaneKeywordsData *pkd = data;
413 if ((type & (NOTIFY_REREAD | NOTIFY_CHANGE | NOTIFY_METADATA)) && fd == pkd->fd) bar_pane_keywords_update(pkd);
416 static gboolean bar_pane_keywords_changed_idle_cb(gpointer data)
418 PaneKeywordsData *pkd = data;
420 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
421 bar_pane_keywords_write(pkd);
422 bar_keyword_tree_sync(pkd);
423 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
428 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data)
430 PaneKeywordsData *pkd = data;
432 if (pkd->idle_id != -1) return;
433 /* higher prio than redraw */
434 pkd->idle_id = g_idle_add_full(G_PRIORITY_HIGH_IDLE, bar_pane_keywords_changed_idle_cb, pkd, NULL);
439 *-------------------------------------------------------------------
441 *-------------------------------------------------------------------
445 static GtkTargetEntry bar_pane_keywords_drag_types[] = {
446 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
447 { "text/plain", 0, TARGET_TEXT_PLAIN }
449 static gint n_keywords_drag_types = 2;
452 static GtkTargetEntry bar_pane_keywords_drop_types[] = {
453 { TARGET_APP_KEYWORD_PATH_STRING, GTK_TARGET_SAME_WIDGET, TARGET_APP_KEYWORD_PATH },
454 { "text/plain", 0, TARGET_TEXT_PLAIN }
456 static gint n_keywords_drop_types = 2;
459 static void bar_pane_keywords_dnd_get(GtkWidget *tree_view, GdkDragContext *context,
460 GtkSelectionData *selection_data, guint info,
461 guint time, gpointer data)
465 GtkTreeIter child_iter;
466 GtkTreeModel *keyword_tree;
468 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
470 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
472 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
473 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
477 case TARGET_APP_KEYWORD_PATH:
479 GList *path = keyword_tree_get_path(keyword_tree, &child_iter);
480 gtk_selection_data_set(selection_data, selection_data->target,
481 8, (gpointer) &path, sizeof(path));
485 case TARGET_TEXT_PLAIN:
488 gchar *name = keyword_get_name(keyword_tree, &child_iter);
489 gtk_selection_data_set_text(selection_data, name, -1);
496 static void bar_pane_keywords_dnd_begin(GtkWidget *tree_view, GdkDragContext *context, gpointer data)
500 GtkTreeIter child_iter;
501 GtkTreeModel *keyword_tree;
504 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_view));
506 if (!gtk_tree_selection_get_selected(sel, &model, &iter)) return;
508 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
509 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
511 name = keyword_get_name(keyword_tree, &child_iter);
513 dnd_set_drag_label(tree_view, context, name);
518 static void bar_pane_keywords_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
523 static gboolean bar_pane_keywords_dnd_can_move(GtkTreeModel *keyword_tree, GtkTreeIter *src_kw_iter, GtkTreeIter *dest_kw_iter)
528 if (dest_kw_iter && keyword_same_parent(keyword_tree, src_kw_iter, dest_kw_iter))
530 return TRUE; /* reordering of siblings is ok */
532 if (!dest_kw_iter && !gtk_tree_model_iter_parent(keyword_tree, &parent, src_kw_iter))
534 return TRUE; /* reordering of top-level siblings is ok */
537 src_name = keyword_get_name(keyword_tree, src_kw_iter);
538 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, src_name, FALSE))
547 static gboolean bar_pane_keywords_dnd_skip_existing(GtkTreeModel *keyword_tree, GtkTreeIter *dest_kw_iter, GList **keywords)
549 /* we have to find at least one keyword that does not already exist as a sibling of dest_kw_iter */
550 GList *work = *keywords;
553 gchar *keyword = work->data;
554 if (keyword_exists(keyword_tree, NULL, dest_kw_iter, keyword, FALSE))
556 GList *next = work->next;
558 *keywords = g_list_delete_link(*keywords, work);
569 static void bar_pane_keywords_dnd_receive(GtkWidget *tree_view, GdkDragContext *context,
571 GtkSelectionData *selection_data, guint info,
572 guint time, gpointer data)
574 PaneKeywordsData *pkd = data;
575 GtkTreePath *tpath = NULL;
576 GtkTreeViewDropPosition pos;
579 GtkTreeModel *keyword_tree;
580 gboolean src_valid = FALSE;
581 GList *new_keywords = NULL;
584 /* iterators for keyword_tree */
585 GtkTreeIter src_kw_iter;
586 GtkTreeIter dest_kw_iter;
587 GtkTreeIter new_kw_iter;
589 g_signal_stop_emission_by_name(tree_view, "drag_data_received");
591 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
592 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
594 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
595 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), NULL, pos);
599 case TARGET_APP_KEYWORD_PATH:
601 GList *path = *(gpointer *)selection_data->data;
602 src_valid = keyword_tree_get_iter(keyword_tree, &src_kw_iter, path);
603 string_list_free(path);
607 new_keywords = string_to_keywords_list((gchar *)selection_data->data);
613 GtkTreeIter dest_iter;
614 gtk_tree_model_get_iter(model, &dest_iter, tpath);
615 gtk_tree_path_free(tpath);
616 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &dest_kw_iter, &dest_iter);
618 if (src_valid && gtk_tree_store_is_ancestor(GTK_TREE_STORE(keyword_tree), &src_kw_iter, &dest_kw_iter))
620 /* can't move to it's own child */
624 if (src_valid && keyword_compare(keyword_tree, &src_kw_iter, &dest_kw_iter) == 0)
626 /* can't move to itself */
630 if ((pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE || pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER) &&
631 !gtk_tree_model_iter_has_child(keyword_tree, &dest_kw_iter))
633 /* the node has no children, all keywords can be added */
634 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &dest_kw_iter);
638 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, &dest_kw_iter))
640 /* the keyword can't be moved if the same name already exist */
643 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, &dest_kw_iter, &new_keywords))
645 /* the keywords can't be added if the same name already exist */
651 case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
652 case GTK_TREE_VIEW_DROP_BEFORE:
653 gtk_tree_store_insert_before(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
655 case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
656 case GTK_TREE_VIEW_DROP_AFTER:
657 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL, &dest_kw_iter);
665 if (src_valid && !bar_pane_keywords_dnd_can_move(keyword_tree, &src_kw_iter, NULL))
667 /* the keyword can't be moved if the same name already exist */
670 if (new_keywords && !bar_pane_keywords_dnd_skip_existing(keyword_tree, NULL, &new_keywords))
672 /* the keywords can't be added if the same name already exist */
675 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &new_kw_iter, NULL);
681 keyword_move_recursive(GTK_TREE_STORE(keyword_tree), &new_kw_iter, &src_kw_iter);
687 gchar *keyword = work->data;
688 keyword_set(GTK_TREE_STORE(keyword_tree), &new_kw_iter, keyword, TRUE);
694 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &new_kw_iter);
698 string_list_free(new_keywords);
699 bar_keyword_tree_sync(pkd);
702 static gint bar_pane_keywords_dnd_motion(GtkWidget *tree_view, GdkDragContext *context,
703 gint x, gint y, guint time, gpointer data)
705 GtkTreePath *tpath = NULL;
706 GtkTreeViewDropPosition pos;
707 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(tree_view), x, y, &tpath, &pos);
711 GtkTreeIter dest_iter;
712 model = gtk_tree_view_get_model(GTK_TREE_VIEW(tree_view));
713 gtk_tree_model_get_iter(model, &dest_iter, tpath);
714 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE && gtk_tree_model_iter_has_child(model, &dest_iter))
715 pos = GTK_TREE_VIEW_DROP_BEFORE;
717 if (pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER && gtk_tree_model_iter_has_child(model, &dest_iter))
718 pos = GTK_TREE_VIEW_DROP_AFTER;
721 gtk_tree_view_set_drag_dest_row(GTK_TREE_VIEW(tree_view), tpath, pos);
722 gtk_tree_path_free(tpath);
724 if (tree_view == gtk_drag_get_source_widget(context))
725 gdk_drag_status(context, GDK_ACTION_MOVE, time);
727 gdk_drag_status(context, GDK_ACTION_COPY, time);
733 *-------------------------------------------------------------------
735 *-------------------------------------------------------------------
738 static void bar_pane_keywords_edit_destroy_cb(GtkWidget *widget, gpointer data)
740 ConfDialogData *cdd = data;
741 gtk_tree_path_free(cdd->click_tpath);
746 static void bar_pane_keywords_edit_cancel_cb(GenericDialog *gd, gpointer data)
751 static void bar_pane_keywords_edit_ok_cb(GenericDialog *gd, gpointer data)
753 ConfDialogData *cdd = data;
754 PaneKeywordsData *pkd = cdd->pkd;
757 GtkTreeModel *keyword_tree;
760 gboolean have_dest = FALSE;
764 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
765 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
767 if (cdd->click_tpath)
770 if (gtk_tree_model_get_iter(model, &iter, cdd->click_tpath))
772 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
777 if (cdd->edit_existing && !have_dest) return;
779 keywords = keyword_list_pull(cdd->edit_widget);
781 if (cdd->edit_existing)
783 if (keywords && keywords->data && /* there should be one keyword */
784 !keyword_exists(keyword_tree, NULL, &kw_iter, keywords->data, TRUE))
786 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, keywords->data, cdd->is_keyword);
791 GList *work = keywords;
796 if (keyword_exists(keyword_tree, NULL, have_dest ? &kw_iter : NULL, work->data, FALSE))
803 gtk_tree_store_insert_after(GTK_TREE_STORE(keyword_tree), &add, NULL, &kw_iter);
807 gtk_tree_store_append(GTK_TREE_STORE(keyword_tree), &add, NULL);
811 keyword_set(GTK_TREE_STORE(keyword_tree), &kw_iter, work->data, cdd->is_keyword);
815 string_list_free(keywords);
818 static void bar_pane_keywords_conf_set_helper(GtkWidget *widget, gpointer data)
820 ConfDialogData *cdd = data;
821 cdd->is_keyword = FALSE;
824 static void bar_pane_keywords_conf_set_kw(GtkWidget *widget, gpointer data)
826 ConfDialogData *cdd = data;
827 cdd->is_keyword = TRUE;
832 static void bar_pane_keywords_edit_dialog(PaneKeywordsData *pkd, gboolean edit_existing)
841 gboolean is_keyword = TRUE;
844 if (edit_existing && pkd->click_tpath)
848 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
850 if (gtk_tree_model_get_iter(model, &iter, pkd->click_tpath))
852 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
853 FILTER_KEYWORD_COLUMN_IS_KEYWORD, &is_keyword, -1);
861 if (edit_existing && !name) return;
864 cdd = g_new0(ConfDialogData, 1);
866 cdd->click_tpath = pkd->click_tpath;
867 pkd->click_tpath = NULL;
868 cdd->is_keyword = is_keyword;
869 cdd->edit_existing = edit_existing;
871 cdd->gd = gd = generic_dialog_new(name ? _("Edit keyword") : _("Add keywords"), "keyword_edit",
873 bar_pane_keywords_edit_cancel_cb, cdd);
874 g_signal_connect(G_OBJECT(gd->dialog), "destroy",
875 G_CALLBACK(bar_pane_keywords_edit_destroy_cb), cdd);
878 generic_dialog_add_message(gd, NULL, name ? _("Configure keyword") : _("Add keyword"), NULL);
880 generic_dialog_add_button(gd, GTK_STOCK_OK, NULL,
881 bar_pane_keywords_edit_ok_cb, TRUE);
883 table = pref_table_new(gd->vbox, 3, 1, FALSE, TRUE);
884 pref_table_label(table, 0, 0, _("Keyword:"), 1.0);
885 cdd->edit_widget = gtk_entry_new();
886 gtk_widget_set_size_request(cdd->edit_widget, 300, -1);
887 if (name) gtk_entry_set_text(GTK_ENTRY(cdd->edit_widget), name);
888 gtk_table_attach_defaults(GTK_TABLE(table), cdd->edit_widget, 1, 2, 0, 1);
889 /* here could eventually be a text view instead of entry */
890 generic_dialog_attach_default(gd, cdd->edit_widget);
891 gtk_widget_show(cdd->edit_widget);
893 group = pref_group_new(gd->vbox, FALSE, _("Keyword type:"), GTK_ORIENTATION_VERTICAL);
895 button = pref_radiobutton_new(group, NULL, _("Active keyword"),
897 G_CALLBACK(bar_pane_keywords_conf_set_kw), cdd);
898 button = pref_radiobutton_new(group, button, _("Helper"),
900 G_CALLBACK(bar_pane_keywords_conf_set_helper), cdd);
903 gtk_widget_show(gd->dialog);
910 *-------------------------------------------------------------------
912 *-------------------------------------------------------------------
915 static void bar_pane_keywords_edit_dialog_cb(GtkWidget *menu_widget, gpointer data)
917 PaneKeywordsData *pkd = data;
918 bar_pane_keywords_edit_dialog(pkd, TRUE);
921 static void bar_pane_keywords_add_dialog_cb(GtkWidget *menu_widget, gpointer data)
923 PaneKeywordsData *pkd = data;
924 bar_pane_keywords_edit_dialog(pkd, FALSE);
927 static void bar_pane_keywords_connect_mark_cb(GtkWidget *menu_widget, gpointer data)
929 PaneKeywordsData *pkd = data;
934 GtkTreeModel *keyword_tree;
937 gint mark = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menu_widget), "mark")) - 1;
939 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
940 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
942 if (!pkd->click_tpath) return;
943 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
945 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
947 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
949 meta_data_connect_mark_with_keyword(keyword_tree, &kw_iter, mark);
951 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
952 // bar_pane_keywords_update(pkd);
956 static void bar_pane_keywords_delete_cb(GtkWidget *menu_widget, gpointer data)
958 PaneKeywordsData *pkd = data;
962 GtkTreeModel *keyword_tree;
965 if (!pkd->click_tpath) return;
967 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
968 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
970 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
971 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
973 keyword_delete(GTK_TREE_STORE(keyword_tree), &kw_iter);
976 static void bar_pane_keywords_hide_cb(GtkWidget *menu_widget, gpointer data)
978 PaneKeywordsData *pkd = data;
982 GtkTreeModel *keyword_tree;
985 if (!pkd->click_tpath) return;
987 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
988 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
990 if (!gtk_tree_model_get_iter(model, &iter, pkd->click_tpath)) return;
991 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &kw_iter, &iter);
993 keyword_hide_in(GTK_TREE_STORE(keyword_tree), &kw_iter, model);
996 static void bar_pane_keywords_show_all_cb(GtkWidget *menu_widget, gpointer data)
998 PaneKeywordsData *pkd = data;
1001 GtkTreeModel *keyword_tree;
1003 pkd->hide_unchecked = FALSE;
1005 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1006 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1008 keyword_show_all_in(GTK_TREE_STORE(keyword_tree), model);
1010 if (!pkd->collapse_unchecked) gtk_tree_view_expand_all(GTK_TREE_VIEW(pkd->keyword_treeview));
1011 bar_keyword_tree_sync(pkd);
1014 static void bar_pane_keywords_expand_checked_cb(GtkWidget *menu_widget, gpointer data)
1016 PaneKeywordsData *pkd = data;
1017 GtkTreeModel *model;
1019 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1020 gtk_tree_model_foreach(model, bar_keyword_tree_expand_if_set_cb, pkd);
1023 static void bar_pane_keywords_collapse_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1025 PaneKeywordsData *pkd = data;
1026 GtkTreeModel *model;
1028 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1029 gtk_tree_model_foreach(model, bar_keyword_tree_collapse_if_unset_cb, pkd);
1032 static void bar_pane_keywords_hide_unchecked_cb(GtkWidget *menu_widget, gpointer data)
1034 PaneKeywordsData *pkd = data;
1035 GtkTreeModel *model;
1037 GtkTreeModel *keyword_tree;
1040 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1041 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
1043 keywords = keyword_list_pull(pkd->keyword_view);
1044 keyword_hide_unset_in(GTK_TREE_STORE(keyword_tree), model, keywords);
1045 string_list_free(keywords);
1046 bar_keyword_tree_sync(pkd);
1049 static void bar_pane_keywords_expand_checked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1051 PaneKeywordsData *pkd = data;
1052 pkd->expand_checked = !pkd->expand_checked;
1053 bar_keyword_tree_sync(pkd);
1056 static void bar_pane_keywords_collapse_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1058 PaneKeywordsData *pkd = data;
1059 pkd->collapse_unchecked = !pkd->collapse_unchecked;
1060 bar_keyword_tree_sync(pkd);
1063 static void bar_pane_keywords_hide_unchecked_toggle_cb(GtkWidget *menu_widget, gpointer data)
1065 PaneKeywordsData *pkd = data;
1066 pkd->hide_unchecked = !pkd->hide_unchecked;
1067 bar_keyword_tree_sync(pkd);
1070 static void bar_pane_keywords_menu_popup(GtkWidget *widget, PaneKeywordsData *pkd, gint x, gint y)
1075 GtkTreeViewDropPosition pos;
1077 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1078 pkd->click_tpath = NULL;
1079 gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(pkd->keyword_treeview), x, y, &pkd->click_tpath, &pos);
1081 menu = popup_menu_short_lived();
1083 if (pkd->click_tpath)
1090 GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
1093 gtk_tree_model_get_iter(model, &iter, pkd->click_tpath);
1096 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_NAME, &name,
1097 FILTER_KEYWORD_COLUMN_MARK, &mark, -1);
1099 text = g_strdup_printf(_("Hide \"%s\""), name);
1100 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_hide_cb), pkd);
1103 submenu = gtk_menu_new();
1104 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
1106 text = g_strdup_printf(_("Mark %d"), i + 1);
1107 item = menu_item_add(submenu, text, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1108 g_object_set_data(G_OBJECT(item), "mark", GINT_TO_POINTER(i + 1));
1111 text = g_strdup_printf(_("Connect \"%s\" to mark"), name);
1112 item = menu_item_add(menu, text, NULL, NULL);
1113 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1116 menu_item_add_divider(menu);
1118 text = g_strdup_printf(_("Edit \"%s\""), name);
1119 menu_item_add_stock(menu, text, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_edit_dialog_cb), pkd);
1121 text = g_strdup_printf(_("Delete \"%s\""), name);
1122 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_delete_cb), pkd);
1126 if (mark && mark[0])
1128 text = g_strdup_printf(_("Disconnect \"%s\" from mark %s"), name, mark);
1129 menu_item_add_stock(menu, text, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_keywords_connect_mark_cb), pkd);
1133 menu_item_add_divider(menu);
1140 menu_item_add(menu, _("Expand checked"), G_CALLBACK(bar_pane_keywords_expand_checked_cb), pkd);
1141 menu_item_add(menu, _("Collapse unchecked"), G_CALLBACK(bar_pane_keywords_collapse_unchecked_cb), pkd);
1142 menu_item_add(menu, _("Hide unchecked"), G_CALLBACK(bar_pane_keywords_hide_unchecked_cb), pkd);
1143 menu_item_add(menu, _("Show all"), G_CALLBACK(bar_pane_keywords_show_all_cb), pkd);
1145 submenu = gtk_menu_new();
1146 item = menu_item_add(menu, _("On any change"), NULL, NULL);
1147 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
1149 menu_item_add_check(submenu, _("Expand checked"), pkd->expand_checked, G_CALLBACK(bar_pane_keywords_expand_checked_toggle_cb), pkd);
1150 menu_item_add_check(submenu, _("Collapse unchecked"), pkd->collapse_unchecked, G_CALLBACK(bar_pane_keywords_collapse_unchecked_toggle_cb), pkd);
1151 menu_item_add_check(submenu, _("Hide unchecked"), pkd->hide_unchecked, G_CALLBACK(bar_pane_keywords_hide_unchecked_toggle_cb), pkd);
1153 menu_item_add_divider(menu);
1155 menu_item_add_stock(menu, _("Add keyword"), GTK_STOCK_EDIT, G_CALLBACK(bar_pane_keywords_add_dialog_cb), pkd);
1157 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 0, GDK_CURRENT_TIME);
1161 static gboolean bar_pane_keywords_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
1163 PaneKeywordsData *pkd = data;
1164 if (bevent->button == MOUSE_BUTTON_RIGHT)
1166 bar_pane_keywords_menu_popup(widget, pkd, bevent->x, bevent->y);
1173 *-------------------------------------------------------------------
1175 *-------------------------------------------------------------------
1178 void bar_pane_keywords_close(GtkWidget *bar)
1180 PaneKeywordsData *pkd;
1182 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
1185 gtk_widget_destroy(pkd->widget);
1188 static void bar_pane_keywords_destroy(GtkWidget *widget, gpointer data)
1190 PaneKeywordsData *pkd = data;
1192 if (pkd->click_tpath) gtk_tree_path_free(pkd->click_tpath);
1193 if (pkd->idle_id != -1) g_source_remove(pkd->idle_id);
1194 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
1196 file_data_unref(pkd->fd);
1203 GtkWidget *bar_pane_keywords_new(const gchar *title, const gchar *key, gboolean expanded)
1205 PaneKeywordsData *pkd;
1207 GtkWidget *scrolled;
1208 GtkTextBuffer *buffer;
1209 GtkTreeModel *store;
1210 GtkTreeViewColumn *column;
1211 GtkCellRenderer *renderer;
1213 pkd = g_new0(PaneKeywordsData, 1);
1215 pkd->pane.pane_set_fd = bar_pane_keywords_set_fd;
1216 pkd->pane.pane_event = bar_pane_keywords_event;
1217 pkd->pane.pane_write_config = bar_pane_keywords_write_config;
1218 pkd->pane.title = bar_pane_expander_title(title);
1220 pkd->pane.expanded = expanded;
1222 pkd->key = g_strdup(key);
1224 pkd->expand_checked = TRUE;
1228 hbox = gtk_hbox_new(FALSE, PREF_PAD_GAP);
1231 g_object_set_data(G_OBJECT(pkd->widget), "pane_data", pkd);
1232 g_signal_connect(G_OBJECT(pkd->widget), "destroy",
1233 G_CALLBACK(bar_pane_keywords_destroy), pkd);
1234 gtk_widget_show(hbox);
1236 scrolled = gtk_scrolled_window_new(NULL, NULL);
1237 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1238 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1239 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1240 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1241 gtk_widget_show(scrolled);
1243 pkd->keyword_view = gtk_text_view_new();
1244 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_view);
1245 g_signal_connect(G_OBJECT(pkd->keyword_view), "populate-popup",
1246 G_CALLBACK(bar_pane_keywords_populate_popup_cb), pkd);
1247 gtk_widget_show(pkd->keyword_view);
1249 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
1250 g_signal_connect(G_OBJECT(buffer), "changed",
1251 G_CALLBACK(bar_pane_keywords_changed), pkd);
1253 scrolled = gtk_scrolled_window_new(NULL, NULL);
1254 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
1255 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
1256 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1257 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
1258 gtk_widget_show(scrolled);
1261 if (!keyword_tree) keyword_tree_new_default();
1263 store = gtk_tree_model_filter_new(GTK_TREE_MODEL(keyword_tree), NULL);
1265 gtk_tree_model_filter_set_modify_func(GTK_TREE_MODEL_FILTER(store),
1266 FILTER_KEYWORD_COLUMN_COUNT,
1267 filter_keyword_column_types,
1268 bar_pane_keywords_filter_modify,
1271 gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(store),
1272 bar_pane_keywords_filter_visible,
1276 pkd->keyword_treeview = gtk_tree_view_new_with_model(store);
1277 g_object_unref(store);
1279 gtk_widget_set_size_request(pkd->keyword_treeview, -1, 400);
1281 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(pkd->keyword_treeview), FALSE);
1283 // gtk_tree_view_set_search_column(GTK_TREE_VIEW(pkd->keyword_treeview), FILTER_KEYWORD_COLUMN_);
1285 column = gtk_tree_view_column_new();
1286 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
1288 renderer = gtk_cell_renderer_text_new();
1289 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1291 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_MARK);
1293 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1295 column = gtk_tree_view_column_new();
1296 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1297 renderer = gtk_cell_renderer_toggle_new();
1298 gtk_tree_view_column_pack_start(column, renderer, FALSE);
1299 gtk_tree_view_column_add_attribute(column, renderer, "active", FILTER_KEYWORD_COLUMN_TOGGLE);
1300 gtk_tree_view_column_add_attribute(column, renderer, "visible", FILTER_KEYWORD_COLUMN_IS_KEYWORD);
1301 g_signal_connect(G_OBJECT(renderer), "toggled",
1302 G_CALLBACK(bar_pane_keywords_keyword_toggle), pkd);
1304 renderer = gtk_cell_renderer_text_new();
1305 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1306 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_NAME);
1308 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1309 gtk_tree_view_set_expander_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
1311 gtk_drag_source_set(pkd->keyword_treeview,
1312 GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
1313 bar_pane_keywords_drag_types, n_keywords_drag_types,
1314 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
1316 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_get",
1317 G_CALLBACK(bar_pane_keywords_dnd_get), pkd);
1319 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_begin",
1320 G_CALLBACK(bar_pane_keywords_dnd_begin), pkd);
1321 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_end",
1322 G_CALLBACK(bar_pane_keywords_dnd_end), pkd);
1324 gtk_drag_dest_set(pkd->keyword_treeview,
1325 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
1326 bar_pane_keywords_drop_types, n_keywords_drop_types,
1327 GDK_ACTION_COPY | GDK_ACTION_MOVE);
1329 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_data_received",
1330 G_CALLBACK(bar_pane_keywords_dnd_receive), pkd);
1332 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "drag_motion",
1333 G_CALLBACK(bar_pane_keywords_dnd_motion), pkd);
1335 g_signal_connect(G_OBJECT(pkd->keyword_treeview), "button_press_event",
1336 G_CALLBACK(bar_pane_keywords_menu_cb), pkd);
1338 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_treeview);
1339 gtk_widget_show(pkd->keyword_treeview);
1341 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
1346 GtkWidget *bar_pane_keywords_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
1348 gchar *title = g_strdup(_("NoName"));
1349 gchar *key = g_strdup(COMMENT_KEY);
1350 gboolean expanded = TRUE;
1352 while (*attribute_names)
1354 const gchar *option = *attribute_names++;
1355 const gchar *value = *attribute_values++;
1357 if (READ_CHAR_FULL("pane.title", title)) continue;
1358 if (READ_CHAR_FULL("key", key)) continue;
1359 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
1362 log_printf("unknown attribute %s = %s\n", option, value);
1365 return bar_pane_keywords_new(title, key, expanded);
1368 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */