Add a contextual menu on bar pane histogram allowing to change channels and mode...
[geeqie.git] / src / bar_histogram.c
1 /*
2  * Geeqie
3  * (C) 2004 John Ellis
4  * Copyright (C) 2008 - 2009 The Geeqie Team
5  *
6  * Author: Vladimir Nadvornik
7  *
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!
11  */
12
13
14 #include "main.h"
15 #include "bar_comment.h"
16
17 #include "bar.h"
18 #include "metadata.h"
19 #include "filedata.h"
20 #include "menu.h"
21 #include "ui_menu.h"
22 #include "ui_misc.h"
23 #include "histogram.h"
24 #include "rcfile.h"
25
26 /*
27  *-------------------------------------------------------------------
28  * keyword / comment utils
29  *-------------------------------------------------------------------
30  */
31
32
33
34 typedef struct _PaneHistogramData PaneHistogramData;
35 struct _PaneHistogramData
36 {
37         PaneData pane;
38         GtkWidget *widget;
39         GtkWidget *drawing_area;
40         Histogram *histogram;
41         gint histogram_width;
42         gint histogram_height;
43         GdkPixbuf *pixbuf;
44         FileData *fd;
45 };
46
47
48 static void bar_pane_histogram_update(PaneHistogramData *phd)
49 {
50         const HistMap *histmap;
51         if (phd->pixbuf) g_object_unref(phd->pixbuf);
52         phd->pixbuf = NULL;
53
54         if (!phd->histogram_width || !phd->histogram_height || !phd->fd) return;
55
56         gtk_widget_queue_draw_area(GTK_WIDGET(phd->drawing_area), 0, 0, phd->histogram_width, phd->histogram_height);
57         
58         histmap = histmap_get(phd->fd);
59         
60         if (!histmap) return;
61         
62         phd->pixbuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, phd->histogram_width, phd->histogram_height);
63         gdk_pixbuf_fill(phd->pixbuf, 0xffffffff);
64         histogram_draw(phd->histogram, histmap, phd->pixbuf, 0, 0, phd->histogram_width, phd->histogram_height);
65 }
66
67
68 static void bar_pane_histogram_set_fd(GtkWidget *pane, FileData *fd)
69 {
70         PaneHistogramData *phd;
71
72         phd = g_object_get_data(G_OBJECT(pane), "pane_data");
73         if (!phd) return;
74
75         file_data_unref(phd->fd);
76         phd->fd = file_data_ref(fd);
77
78         bar_pane_histogram_update(phd);
79 }
80
81 static void bar_pane_histogram_write_config(GtkWidget *pane, GString *outstr, gint indent)
82 {
83         PaneHistogramData *phd;
84
85         phd = g_object_get_data(G_OBJECT(pane), "pane_data");
86         if (!phd) return;
87
88         WRITE_STRING("<pane_histogram\n");
89         indent++;
90         WRITE_CHAR(*phd, pane.title);
91         WRITE_BOOL(*phd, pane.expanded);
92         indent--;
93         WRITE_STRING("/>\n");
94 }
95
96
97 static void bar_pane_histogram_notify_cb(FileData *fd, NotifyType type, gpointer data)
98 {
99         PaneHistogramData *phd = data;
100         if (fd == phd->fd) bar_pane_histogram_update(phd);
101 }
102
103 static gboolean bar_pane_histogram_expose_event_cb (GtkWidget *widget, GdkEventExpose *event, gpointer data)
104 {
105         PaneHistogramData *phd = data;
106         if (!phd || !phd->pixbuf) return TRUE;
107         
108         gdk_draw_pixbuf(widget->window,
109                         widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
110                         phd->pixbuf,
111                         0, 0,
112                         0, 0,
113                         -1, -1,
114                         GDK_RGB_DITHER_NORMAL, 0, 0);
115         return TRUE;
116 }
117
118 static void bar_pane_histogram_size_cb(GtkWidget *widget, GtkAllocation *allocation, gpointer data)
119 {
120         PaneHistogramData *phd = data;
121
122         phd->histogram_width = allocation->width;
123         phd->histogram_height = allocation->height;
124         bar_pane_histogram_update(phd);
125 }
126
127 static void bar_pane_histogram_close(GtkWidget *pane)
128 {
129         PaneHistogramData *phd;
130
131         phd = g_object_get_data(G_OBJECT(pane), "pane_data");
132         if (!phd) return;
133
134         gtk_widget_destroy(phd->widget);
135 }
136
137 static void bar_pane_histogram_destroy(GtkWidget *widget, gpointer data)
138 {
139         PaneHistogramData *phd = data;
140
141         file_data_unregister_notify_func(bar_pane_histogram_notify_cb, phd);
142
143         file_data_unref(phd->fd);
144         g_free(phd->pane.title);
145         histogram_free(phd->histogram);
146         if (phd->pixbuf) g_object_unref(phd->pixbuf);
147
148         g_free(phd);
149 }
150
151 static void bar_pane_histogram_popup_channels_cb(GtkWidget *widget, gpointer data)
152 {
153         PaneHistogramData *phd;
154         gint channel;
155
156         phd = submenu_item_get_data(widget);
157
158         if (!phd) return;
159
160         channel = GPOINTER_TO_INT(data);
161         histogram_set_channel(phd->histogram, channel);
162         bar_pane_histogram_update(phd);
163 }
164
165 static void bar_pane_histogram_popup_logmode_cb(GtkWidget *widget, gpointer data)
166 {
167         PaneHistogramData *phd;
168         gint logmode;
169
170         phd = submenu_item_get_data(widget);
171
172         if (!phd) return;
173
174         logmode = GPOINTER_TO_INT(data);
175         histogram_set_mode(phd->histogram, logmode);
176         bar_pane_histogram_update(phd);
177 }
178
179 static GtkWidget *bar_pane_histogram_add_radio(GtkWidget *menu, GtkWidget *parent,
180                                         const gchar *label,
181                                         GCallback func, gint value,
182                                         gboolean show_current, gint show_value)
183 {
184         GtkWidget *item;
185
186         if (show_current)
187                 {
188                 item = menu_item_add_radio(menu, parent,
189                                            label, (value == show_value),
190                                            func, GINT_TO_POINTER((gint)value));
191                 }
192         else
193                 {
194                 item = menu_item_add(menu, label,
195                                      func, GINT_TO_POINTER((gint)value));
196                 }
197
198         return item;
199 }
200
201 GtkWidget *bar_pane_histogram_add_channels(GtkWidget *menu, GCallback func, gpointer data,
202                                            gboolean show_current, gint value)
203 {
204         GtkWidget *submenu;
205         GtkWidget *parent;
206
207         submenu = gtk_menu_new();
208         g_object_set_data(G_OBJECT(submenu), "submenu_data", data);
209
210         parent = bar_pane_histogram_add_radio(submenu, NULL, _("_Red"), func, HCHAN_R, show_current, value);
211         bar_pane_histogram_add_radio(submenu, parent, _("_Green"), func, HCHAN_G, show_current, value);
212         bar_pane_histogram_add_radio(submenu, parent, _("_Blue"),func, HCHAN_B, show_current, value);
213         bar_pane_histogram_add_radio(submenu, parent, _("_RGB"),func, HCHAN_RGB, show_current, value);
214         bar_pane_histogram_add_radio(submenu, parent, _("_Value"),func, HCHAN_MAX, show_current, value);
215
216         if (menu)
217                 {
218                 GtkWidget *item;
219
220                 item = menu_item_add(menu, _("Channels"), NULL, NULL);
221                 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
222                 return item;
223                 }
224
225         return submenu;
226 }
227 GtkWidget *bar_pane_histogram_add_logmode(GtkWidget *menu, GCallback func, gpointer data,
228                                            gboolean show_current, gint value)
229 {
230         GtkWidget *submenu;
231         GtkWidget *parent;
232
233         submenu = gtk_menu_new();
234         g_object_set_data(G_OBJECT(submenu), "submenu_data", data);
235
236         parent = bar_pane_histogram_add_radio(submenu, NULL, _("_Linear"), func, 0, show_current, value);
237         bar_pane_histogram_add_radio(submenu, parent, _("Lo_garithmical"), func, 1, show_current, value);
238
239         if (menu)
240                 {
241                 GtkWidget *item;
242
243                 item = menu_item_add(menu, _("Mode"), NULL, NULL);
244                 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
245                 return item;
246                 }
247
248         return submenu;
249 }
250
251
252 static GtkWidget *bar_pane_histogram_menu(PaneHistogramData *phd)
253 {
254         GtkWidget *menu;
255         static gboolean show_current = FALSE; /* FIXME: TRUE -> buggy behavior */
256
257         menu = popup_menu_short_lived();
258         bar_pane_histogram_add_channels(menu, G_CALLBACK(bar_pane_histogram_popup_channels_cb), phd,
259                                         show_current, histogram_get_channel(phd->histogram));
260         bar_pane_histogram_add_logmode(menu, G_CALLBACK(bar_pane_histogram_popup_logmode_cb), phd,
261                                        show_current, histogram_get_mode(phd->histogram));
262
263         return menu;
264 }
265
266 static gboolean bar_pane_histogram_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
267 {
268         PaneHistogramData *phd = data;
269
270         if (bevent->button == MOUSE_BUTTON_RIGHT)
271                 {
272                 GtkWidget *menu;
273
274                 menu = bar_pane_histogram_menu(phd);
275                 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, bevent->button, bevent->time);
276                 return TRUE;
277         }
278
279         return FALSE;
280 }
281
282
283 GtkWidget *bar_pane_histogram_new(const gchar *title, gint height, gint expanded)
284 {
285         PaneHistogramData *phd;
286
287         phd = g_new0(PaneHistogramData, 1);
288         
289         phd->pane.pane_set_fd = bar_pane_histogram_set_fd;
290         phd->pane.pane_write_config = bar_pane_histogram_write_config;
291         phd->pane.title = g_strdup(title);
292         phd->pane.expanded = expanded;
293         
294         phd->histogram = histogram_new();
295
296         histogram_set_channel(phd->histogram, HCHAN_RGB);
297         histogram_set_mode(phd->histogram, 0);
298
299         phd->widget = gtk_vbox_new(FALSE, PREF_PAD_GAP);
300
301         g_object_set_data(G_OBJECT(phd->widget), "pane_data", phd);
302         g_signal_connect(G_OBJECT(phd->widget), "destroy",
303                          G_CALLBACK(bar_pane_histogram_destroy), phd);
304         
305
306         gtk_widget_set_size_request(GTK_WIDGET(phd->widget), -1, height);
307
308         phd->drawing_area = gtk_drawing_area_new();
309         g_signal_connect_after(G_OBJECT(phd->drawing_area), "size_allocate",
310                                G_CALLBACK(bar_pane_histogram_size_cb), phd);
311
312         g_signal_connect(G_OBJECT(phd->drawing_area), "expose_event",  
313                          G_CALLBACK(bar_pane_histogram_expose_event_cb), phd);
314                          
315         gtk_box_pack_start(GTK_BOX(phd->widget), phd->drawing_area, TRUE, TRUE, 0);
316         gtk_widget_show(phd->drawing_area);
317         gtk_widget_add_events(phd->drawing_area, GDK_BUTTON_PRESS_MASK);
318
319         g_signal_connect(G_OBJECT(phd->drawing_area), "button_press_event", G_CALLBACK(bar_pane_histogram_press_cb), phd);
320
321         gtk_widget_show(phd->widget);
322
323         file_data_register_notify_func(bar_pane_histogram_notify_cb, phd, NOTIFY_PRIORITY_LOW);
324
325         return phd->widget;
326 }
327
328 GtkWidget *bar_pane_histogram_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
329 {
330         gchar *title = g_strdup(_("NoName"));
331         gboolean expanded = TRUE;
332         gint height = 80;
333
334         while (*attribute_names)
335                 {
336                 const gchar *option = *attribute_names++;
337                 const gchar *value = *attribute_values++;
338
339                 if (READ_CHAR_FULL("pane.title", title)) continue;
340                 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
341                 
342
343                 DEBUG_1("unknown attribute %s = %s", option, value);
344                 }
345         
346         return bar_pane_histogram_new(title, height, expanded);
347 }
348
349
350 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */