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"
31 static const gchar *keyword_favorite_defaults[] = {
43 static void bar_pane_keywords_keyword_update_all(void);
44 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data);
47 *-------------------------------------------------------------------
48 * keyword / comment utils
49 *-------------------------------------------------------------------
53 GList *keyword_list_pull(GtkWidget *text_widget)
58 text = text_widget_text_pull(text_widget);
59 list = string_to_keywords_list(text);
66 static void keyword_list_push(GtkWidget *textview, GList *list)
68 GtkTextBuffer *buffer;
69 GtkTextIter start, end;
71 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(textview));
72 gtk_text_buffer_get_bounds(buffer, &start, &end);
73 gtk_text_buffer_delete(buffer, &start, &end);
77 const gchar *word = list->data;
80 gtk_text_buffer_get_end_iter(buffer, &iter);
81 if (word) gtk_text_buffer_insert(buffer, &iter, word, -1);
82 gtk_text_buffer_get_end_iter(buffer, &iter);
83 gtk_text_buffer_insert(buffer, &iter, "\n", -1);
91 *-------------------------------------------------------------------
93 *-------------------------------------------------------------------
98 FILTER_KEYWORD_COLUMN_TOGGLE = 0,
99 FILTER_KEYWORD_COLUMN_MARK,
100 FILTER_KEYWORD_COLUMN_NAME,
101 FILTER_KEYWORD_COLUMN_IS_KEYWORD,
102 FILTER_KEYWORD_COLUMN_COUNT
105 static GType filter_keyword_column_types[] = {G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN};
107 typedef struct _PaneKeywordsData PaneKeywordsData;
108 struct _PaneKeywordsData
113 GtkWidget *keyword_view;
114 GtkWidget *keyword_treeview;
121 static GList *bar_list = NULL;
124 static void bar_pane_keywords_write(PaneKeywordsData *pkd)
128 if (!pkd->fd) return;
130 list = keyword_list_pull(pkd->keyword_view);
132 metadata_write_list(pkd->fd, KEYWORD_KEY, list);
134 string_list_free(list);
137 static gchar *bar_pane_keywords_get_mark_text(const gchar *key)
140 static gchar buf[10];
142 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
144 FileDataGetMarkFunc get_mark_func;
145 FileDataSetMarkFunc set_mark_func;
147 file_data_get_registered_mark_func(i, &get_mark_func, &set_mark_func, &data);
148 if (get_mark_func == meta_data_get_keyword_mark && strcmp(data, key) == 0)
150 g_sprintf(buf, " %d ", i + 1);
157 gboolean bar_keyword_tree_expand_if_set(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
159 PaneKeywordsData *pkd = data;
162 gtk_tree_model_get(model, iter, FILTER_KEYWORD_COLUMN_TOGGLE, &set, -1);
164 if (set && !gtk_tree_view_row_expanded(GTK_TREE_VIEW(pkd->keyword_treeview), path))
166 gtk_tree_view_expand_to_path(GTK_TREE_VIEW(pkd->keyword_treeview), path);
171 static void bar_keyword_tree_sync(PaneKeywordsData *pkd)
173 GtkTreeModelFilter *store;
175 store = GTK_TREE_MODEL_FILTER(gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview)));
177 gtk_tree_model_filter_refilter(store);
178 gtk_tree_model_foreach(GTK_TREE_MODEL(store), bar_keyword_tree_expand_if_set, pkd);
182 static void bar_pane_keywords_keyword_update_all(void)
189 PaneKeywordsData *pkd;
195 bar_keyword_tree_sync(pkd);
199 static void bar_pane_keywords_update(PaneKeywordsData *pkd)
201 GList *keywords = NULL;
202 GtkTextBuffer *keyword_buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
204 g_signal_handlers_block_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
206 keywords = metadata_read_list(pkd->fd, KEYWORD_KEY, METADATA_PLAIN);
207 keyword_list_push(pkd->keyword_view, keywords);
208 bar_keyword_tree_sync(pkd);
209 string_list_free(keywords);
211 g_signal_handlers_unblock_by_func(keyword_buffer, bar_pane_keywords_changed, pkd);
215 void bar_pane_keywords_set_fd(GtkWidget *pane, FileData *fd)
217 PaneKeywordsData *pkd;
219 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
222 file_data_unref(pkd->fd);
223 pkd->fd = file_data_ref(fd);
225 bar_pane_keywords_update(pkd);
228 static void bar_pane_keywords_write_config(GtkWidget *pane, GString *outstr, gint indent)
230 PaneKeywordsData *pkd;
232 pkd = g_object_get_data(G_OBJECT(pane), "pane_data");
235 WRITE_STRING("<pane_keywords\n");
237 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(pkd->pane.title)));
238 WRITE_BOOL(*pkd, pane.expanded);
239 WRITE_CHAR(*pkd, key);
241 WRITE_STRING("/>\n");
244 gint bar_pane_keywords_event(GtkWidget *bar, GdkEvent *event)
246 PaneKeywordsData *pkd;
248 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
249 if (!pkd) return FALSE;
251 if (GTK_WIDGET_HAS_FOCUS(pkd->keyword_view)) return gtk_widget_event(pkd->keyword_view, event);
256 static void bar_pane_keywords_keyword_toggle(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
258 PaneKeywordsData *pkd = data;
264 GtkTreeIter child_iter;
265 GtkTreeModel *keyword_tree;
267 model = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
269 tpath = gtk_tree_path_new_from_string(path);
270 gtk_tree_model_get_iter(model, &iter, tpath);
271 gtk_tree_path_free(tpath);
273 gtk_tree_model_get(model, &iter, FILTER_KEYWORD_COLUMN_TOGGLE, &active, -1);
277 keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
278 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, &iter);
280 list = keyword_list_pull(pkd->keyword_view);
282 keyword_tree_set(keyword_tree, &child_iter, &list);
284 keyword_tree_reset(keyword_tree, &child_iter, &list);
286 keyword_list_push(pkd->keyword_view, list);
287 string_list_free(list);
288 bar_keyword_tree_sync(pkd);
291 void bar_pane_keywords_filter_modify(GtkTreeModel *model, GtkTreeIter *iter, GValue *value, gint column, gpointer data)
293 PaneKeywordsData *pkd = data;
294 GtkTreeModel *keyword_tree = gtk_tree_model_filter_get_model(GTK_TREE_MODEL_FILTER(model));
295 GtkTreeIter child_iter;
297 gtk_tree_model_filter_convert_iter_to_child_iter(GTK_TREE_MODEL_FILTER(model), &child_iter, iter);
299 memset(value, 0, sizeof (GValue));
303 case FILTER_KEYWORD_COLUMN_TOGGLE:
305 GList *keywords = keyword_list_pull(pkd->keyword_view);
306 gboolean set = keyword_tree_is_set(keyword_tree, &child_iter, keywords);
307 string_list_free(keywords);
309 g_value_init(value, G_TYPE_BOOLEAN);
310 g_value_set_boolean(value, set);
313 case FILTER_KEYWORD_COLUMN_MARK:
314 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_MARK, value);
316 case FILTER_KEYWORD_COLUMN_NAME:
317 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_NAME, value);
319 case FILTER_KEYWORD_COLUMN_IS_KEYWORD:
320 gtk_tree_model_get_value(keyword_tree, &child_iter, KEYWORD_COLUMN_IS_KEYWORD, value);
327 static void bar_pane_keywords_set_selection(PaneKeywordsData *pkd, gboolean append)
329 GList *keywords = NULL;
333 keywords = keyword_list_pull(pkd->keyword_view);
335 list = layout_selection_list(pkd->pane.lw);
339 FileData *fd = work->data;
344 metadata_append_list(fd, KEYWORD_KEY, keywords);
348 metadata_write_list(fd, KEYWORD_KEY, keywords);
353 string_list_free(keywords);
356 static void bar_pane_keywords_sel_add_cb(GtkWidget *button, gpointer data)
358 PaneKeywordsData *pkd = data;
360 bar_pane_keywords_set_selection(pkd, TRUE);
363 static void bar_pane_keywords_sel_replace_cb(GtkWidget *button, gpointer data)
365 PaneKeywordsData *pkd = data;
367 bar_pane_keywords_set_selection(pkd, FALSE);
370 static void bar_pane_keywords_populate_popup_cb(GtkTextView *textview, GtkMenu *menu, gpointer data)
372 PaneKeywordsData *pkd = data;
374 menu_item_add_divider(GTK_WIDGET(menu));
375 menu_item_add_stock(GTK_WIDGET(menu), _("Add keywords to selected files"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_keywords_sel_add_cb), pkd);
376 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);
380 static void bar_pane_keywords_notify_cb(FileData *fd, NotifyType type, gpointer data)
382 PaneKeywordsData *pkd = data;
383 if (fd == pkd->fd) bar_pane_keywords_update(pkd);
386 static void bar_pane_keywords_changed(GtkTextBuffer *buffer, gpointer data)
388 PaneKeywordsData *pkd = data;
390 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
391 bar_pane_keywords_write(pkd);
392 bar_keyword_tree_sync(pkd);
393 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
396 static void bar_pane_keywords_mark_edited(GtkCellRendererText *cell, const gchar *path, const gchar *text, gpointer data)
398 /* PaneKeywordsData *pkd = data;
404 FileDataGetMarkFunc get_mark_func;
405 FileDataSetMarkFunc set_mark_func;
406 gpointer mark_func_data;
408 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
410 store = gtk_tree_view_get_model(GTK_TREE_VIEW(pkd->keyword_treeview));
412 tpath = gtk_tree_path_new_from_string(path);
413 gtk_tree_model_get_iter(store, &iter, tpath);
414 gtk_tree_path_free(tpath);
416 gtk_tree_model_get(store, &iter, FILTER_KEYWORD_COLUMN_TEXT, &key, -1);
418 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
420 file_data_get_registered_mark_func(i, &get_mark_func, &set_mark_func, &mark_func_data);
421 if (get_mark_func == meta_data_get_keyword_mark && strcmp(mark_func_data, key) == 0)
423 g_free(mark_func_data);
424 file_data_register_mark_func(i, NULL, NULL, NULL);
428 if (sscanf(text, " %d ", &i) &&i >=1 && i <= FILEDATA_MARKS_SIZE)
431 file_data_get_registered_mark_func(i, &get_mark_func, &set_mark_func, &mark_func_data);
432 if (get_mark_func == meta_data_get_keyword_mark && mark_func_data) g_free(mark_func_data);
433 file_data_register_mark_func(i, meta_data_get_keyword_mark, meta_data_set_keyword_mark, g_strdup(key));
438 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
439 bar_pane_keywords_update(pkd);
443 void bar_pane_keywords_close(GtkWidget *bar)
445 PaneKeywordsData *pkd;
447 pkd = g_object_get_data(G_OBJECT(bar), "pane_data");
450 gtk_widget_destroy(pkd->widget);
453 static void bar_pane_keywords_destroy(GtkWidget *widget, gpointer data)
455 PaneKeywordsData *pkd = data;
457 file_data_unregister_notify_func(bar_pane_keywords_notify_cb, pkd);
459 file_data_unref(pkd->fd);
465 static GtkTreeModel *create_marks_list(void)
471 /* create list store */
472 model = gtk_list_store_new(1, G_TYPE_STRING);
473 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
476 g_sprintf(str, " %d ", i + 1);
477 gtk_list_store_append(model, &iter);
478 gtk_list_store_set(model, &iter, 0, str, -1);
480 gtk_list_store_append(model, &iter);
481 gtk_list_store_set(model, &iter, 0, " ... ", -1);
482 return GTK_TREE_MODEL(model);
486 GtkWidget *bar_pane_keywords_new(const gchar *title, const gchar *key, gboolean expanded)
488 PaneKeywordsData *pkd;
491 GtkTextBuffer *buffer;
493 GtkTreeViewColumn *column;
494 GtkCellRenderer *renderer;
496 pkd = g_new0(PaneKeywordsData, 1);
498 pkd->pane.pane_set_fd = bar_pane_keywords_set_fd;
499 pkd->pane.pane_event = bar_pane_keywords_event;
500 pkd->pane.pane_write_config = bar_pane_keywords_write_config;
501 pkd->pane.title = bar_pane_expander_title(title);
503 pkd->pane.expanded = expanded;
505 pkd->key = g_strdup(key);
508 hbox = gtk_hbox_new(FALSE, PREF_PAD_GAP);
511 g_object_set_data(G_OBJECT(pkd->widget), "pane_data", pkd);
512 g_signal_connect(G_OBJECT(pkd->widget), "destroy",
513 G_CALLBACK(bar_pane_keywords_destroy), pkd);
514 gtk_widget_show(hbox);
516 scrolled = gtk_scrolled_window_new(NULL, NULL);
517 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
518 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
519 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
520 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
521 gtk_widget_show(scrolled);
523 pkd->keyword_view = gtk_text_view_new();
524 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_view);
525 g_signal_connect(G_OBJECT(pkd->keyword_view), "populate-popup",
526 G_CALLBACK(bar_pane_keywords_populate_popup_cb), pkd);
527 gtk_widget_show(pkd->keyword_view);
529 buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pkd->keyword_view));
530 g_signal_connect(G_OBJECT(buffer), "changed",
531 G_CALLBACK(bar_pane_keywords_changed), pkd);
533 scrolled = gtk_scrolled_window_new(NULL, NULL);
534 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
535 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
536 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
537 gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 0);
538 gtk_widget_show(scrolled);
541 if (!keyword_tree) keyword_tree_new_default();
543 store = gtk_tree_model_filter_new(GTK_TREE_MODEL(keyword_tree), NULL);
545 gtk_tree_model_filter_set_modify_func(GTK_TREE_MODEL_FILTER(store),
546 FILTER_KEYWORD_COLUMN_COUNT,
547 filter_keyword_column_types,
548 bar_pane_keywords_filter_modify,
552 pkd->keyword_treeview = gtk_tree_view_new_with_model(store);
553 g_object_unref(store);
555 gtk_widget_set_size_request(pkd->keyword_treeview, -1, 400);
557 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(pkd->keyword_treeview), FALSE);
559 // gtk_tree_view_set_search_column(GTK_TREE_VIEW(pkd->keyword_treeview), FILTER_KEYWORD_COLUMN_);
561 column = gtk_tree_view_column_new();
562 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
564 renderer = gtk_cell_renderer_combo_new();
565 g_object_set(G_OBJECT(renderer), "editable", (gboolean)TRUE,
566 "model", create_marks_list(),
571 gtk_tree_view_column_pack_start(column, renderer, TRUE);
572 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_MARK);
573 g_signal_connect(renderer, "edited",
574 G_CALLBACK (bar_pane_keywords_mark_edited), pkd);
575 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
578 column = gtk_tree_view_column_new();
579 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
580 renderer = gtk_cell_renderer_toggle_new();
581 gtk_tree_view_column_pack_start(column, renderer, FALSE);
582 gtk_tree_view_column_add_attribute(column, renderer, "active", FILTER_KEYWORD_COLUMN_TOGGLE);
583 gtk_tree_view_column_add_attribute(column, renderer, "visible", FILTER_KEYWORD_COLUMN_IS_KEYWORD);
584 g_signal_connect(G_OBJECT(renderer), "toggled",
585 G_CALLBACK(bar_pane_keywords_keyword_toggle), pkd);
587 renderer = gtk_cell_renderer_text_new();
588 gtk_tree_view_column_pack_start(column, renderer, TRUE);
589 gtk_tree_view_column_add_attribute(column, renderer, "text", FILTER_KEYWORD_COLUMN_NAME);
591 gtk_tree_view_append_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
592 gtk_tree_view_set_expander_column(GTK_TREE_VIEW(pkd->keyword_treeview), column);
594 gtk_container_add(GTK_CONTAINER(scrolled), pkd->keyword_treeview);
595 gtk_widget_show(pkd->keyword_treeview);
597 file_data_register_notify_func(bar_pane_keywords_notify_cb, pkd, NOTIFY_PRIORITY_LOW);
602 GtkWidget *bar_pane_keywords_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
604 gchar *title = g_strdup(_("NoName"));
605 gchar *key = g_strdup(COMMENT_KEY);
606 gboolean expanded = TRUE;
608 while (*attribute_names)
610 const gchar *option = *attribute_names++;
611 const gchar *value = *attribute_values++;
613 if (READ_CHAR_FULL("pane.title", title)) continue;
614 if (READ_CHAR_FULL("key", key)) continue;
615 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
618 DEBUG_1("unknown attribute %s = %s", option, value);
621 return bar_pane_keywords_new(title, key, expanded);
624 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */