comparison finch/libgnt/gntfilesel.c @ 15847:a2ab257116ce

File selector dialog. Still in an experimental state. When properly complete, this can be used for file/folder-request etc.
author Sadrul Habib Chowdhury <imadil@gmail.com>
date Tue, 20 Mar 2007 01:16:35 +0000
parents
children 3da9d5da9054
comparison
equal deleted inserted replaced
15841:c4f348500fa7 15847:a2ab257116ce
1 #include "gntbutton.h"
2 #include "gntentry.h"
3 #include "gntfilesel.h"
4 #include "gntlabel.h"
5 #include "gntmarshal.h"
6 #include "gntstyle.h"
7 #include "gnttree.h"
8
9 #include <string.h>
10 #include <sys/types.h>
11 #include <sys/stat.h>
12 #include <unistd.h>
13
14 #include <glob.h>
15
16 enum
17 {
18 SIG_FILE_SELECTED,
19 SIGS
20 };
21
22 static GntWindowClass *parent_class = NULL;
23 static guint signals[SIGS] = { 0 };
24 static void (*orig_map)(GntWidget *widget);
25
26 static void
27 gnt_file_sel_destroy(GntWidget *widget)
28 {
29 GntFileSel *sel = GNT_FILE_SEL(widget);
30 g_free(sel->current);
31 }
32
33 static char *
34 process_path(const char *path)
35 {
36 char **splits = NULL;
37 int i, j;
38 char *str, *ret;
39
40 splits = g_strsplit(path, G_DIR_SEPARATOR_S, -1);
41 for (i = 0, j = 0; splits[i]; i++) {
42 if (strcmp(splits[i], ".") == 0) {
43 } else if (strcmp(splits[i], "..") == 0) {
44 if (j)
45 j--;
46 } else {
47 if (i != j) {
48 g_free(splits[j]);
49 splits[j] = splits[i];
50 splits[i] = NULL;
51 }
52 j++;
53 }
54 }
55 g_free(splits[j]);
56 splits[j] = NULL;
57 str = g_build_pathv(G_DIR_SEPARATOR_S, splits);
58 ret = g_strdup_printf(G_DIR_SEPARATOR_S "%s", str);
59 g_free(str);
60 g_strfreev(splits);
61 return ret;
62 }
63
64 static void
65 update_location(GntFileSel *sel)
66 {
67 char *old;
68 const char *tmp;
69 tmp = (const char*)gnt_tree_get_selection_data(GNT_TREE(sel->files));
70 old = g_strdup_printf("%s%s%s", sel->current, sel->current[1] ? G_DIR_SEPARATOR_S : "", tmp ? tmp : "");
71 gnt_entry_set_text(GNT_ENTRY(sel->location), old);
72 g_free(old);
73 }
74
75 static gboolean
76 location_changed(GntFileSel *sel, GError **err)
77 {
78 GDir *dir;
79 const char *str;
80
81 gnt_tree_remove_all(GNT_TREE(sel->dirs));
82 gnt_tree_remove_all(GNT_TREE(sel->files));
83 gnt_entry_set_text(GNT_ENTRY(sel->location), NULL);
84 if (sel->current == NULL) {
85 if (GNT_WIDGET_IS_FLAG_SET(GNT_WIDGET(sel), GNT_WIDGET_MAPPED))
86 gnt_widget_draw(GNT_WIDGET(sel));
87 return TRUE;
88 }
89
90 dir = g_dir_open(sel->current, 0, err);
91 if (dir == NULL || *err) {
92 g_printerr("GntFileSel: error opening location %s (%s)\n",
93 sel->current, *err ? (*err)->message : "reason unknown");
94 return FALSE;
95 }
96
97 if (*sel->current != '\0' && strcmp(sel->current, G_DIR_SEPARATOR_S))
98 gnt_tree_add_row_after(GNT_TREE(sel->dirs), g_strdup(".."),
99 gnt_tree_create_row(GNT_TREE(sel->dirs), ".."), NULL, NULL);
100
101 while ((str = g_dir_read_name(dir)) != NULL) {
102 char *fp = g_build_filename(sel->current, str, NULL);
103 struct stat st;
104
105 if (stat(fp, &st)) {
106 g_printerr("Error stating location %s\n", fp);
107 } else {
108 if (S_ISDIR(st.st_mode))
109 gnt_tree_add_row_after(GNT_TREE(sel->dirs), g_strdup(str),
110 gnt_tree_create_row(GNT_TREE(sel->dirs), str), NULL, NULL);
111 else {
112 char size[128];
113 snprintf(size, sizeof(size), "%ld", (long)st.st_size);
114
115 gnt_tree_add_row_after(GNT_TREE(sel->files), g_strdup(str),
116 gnt_tree_create_row(GNT_TREE(sel->files), str, size, ""), NULL, NULL);
117 }
118 }
119 g_free(fp);
120 }
121 if (GNT_WIDGET_IS_FLAG_SET(GNT_WIDGET(sel), GNT_WIDGET_MAPPED))
122 gnt_widget_draw(GNT_WIDGET(sel));
123 return TRUE;
124 }
125
126 static gboolean
127 dir_key_pressed(GntTree *tree, const char *key, GntFileSel *sel)
128 {
129 if (strcmp(key, "\r") == 0) {
130 /* XXX: if we are moving up the tree, make sure the current node is selected after the redraw */
131 char *str = g_strdup(gnt_tree_get_selection_data(tree));
132 char *path = g_build_filename(sel->current, str, NULL);
133 char *dir = g_path_get_basename(sel->current);
134 if (!gnt_file_sel_set_current_location(sel, path)) {
135 gnt_tree_set_selected(tree, str);
136 } else if (strcmp(str, "..") == 0) {
137 gnt_tree_set_selected(tree, dir);
138 }
139 g_free(dir);
140 g_free(str);
141 g_free(path);
142 return TRUE;
143 }
144 return FALSE;
145 }
146
147 static gboolean
148 location_key_pressed(GntTree *tree, const char *key, GntFileSel *sel)
149 {
150 if (strcmp(key, "\r") == 0) {
151 int count;
152 glob_t gl;
153 char *path;
154 char *str;
155 struct stat st;
156 int glob_ret;
157
158 str = (char*)gnt_entry_get_text(GNT_ENTRY(sel->location));
159 if (*str == G_DIR_SEPARATOR)
160 path = g_strdup(str);
161 else
162 path = g_strdup_printf("%s" G_DIR_SEPARATOR_S "%s", sel->current, str);
163 str = process_path(path);
164 g_free(path);
165 path = str;
166
167 if (!stat(path, &st)) {
168 if (S_ISDIR(st.st_mode)) {
169 gnt_file_sel_set_current_location(sel, path);
170 goto success;
171 }
172 }
173
174 glob_ret = glob(path, GLOB_MARK, NULL, &gl);
175 if (!glob_ret) { /* XXX: do something with the return value */
176 char *loc = g_path_get_dirname(gl.gl_pathv[0]);
177
178 stat(gl.gl_pathv[0], &st);
179 gnt_file_sel_set_current_location(sel, loc); /* XXX: check the return value */
180 g_free(loc);
181 if (!S_ISDIR(st.st_mode)) {
182 gnt_tree_remove_all(GNT_TREE(sel->files));
183 for (count = 0; count < gl.gl_pathc; count++) {
184 char *tmp = process_path(gl.gl_pathv[count]);
185 loc = g_path_get_dirname(tmp);
186 if (g_utf8_collate(sel->current, loc) == 0) {
187 char *base = g_path_get_basename(tmp);
188 char size[128];
189 snprintf(size, sizeof(size), "%ld", (long)st.st_size);
190 gnt_tree_add_row_after(GNT_TREE(sel->files), base,
191 gnt_tree_create_row(GNT_TREE(sel->files), base, size, ""), NULL, NULL);
192 }
193 g_free(loc);
194 g_free(tmp);
195 }
196 gnt_widget_draw(sel->files);
197 }
198 } else {
199 gnt_tree_remove_all(GNT_TREE(sel->files));
200 gnt_widget_draw(sel->files);
201 }
202 globfree(&gl);
203 success:
204 g_free(path);
205 return TRUE;
206 }
207 return FALSE;
208 }
209
210 static void
211 file_sel_changed(GntWidget *widget, gpointer old, gpointer current, GntFileSel *sel)
212 {
213 update_location(sel);
214 }
215
216 static void
217 gnt_file_sel_map(GntWidget *widget)
218 {
219 orig_map(widget);
220 update_location(GNT_FILE_SEL(widget));
221 }
222
223 static void
224 gnt_file_sel_class_init(GntFileSelClass *klass)
225 {
226 GntWidgetClass *kl = GNT_WIDGET_CLASS(klass);
227 parent_class = GNT_WINDOW_CLASS(klass);
228 kl->destroy = gnt_file_sel_destroy;
229 orig_map = kl->map;
230 kl->map = gnt_file_sel_map;
231
232 signals[SIG_FILE_SELECTED] =
233 g_signal_new("file_selected",
234 G_TYPE_FROM_CLASS(klass),
235 G_SIGNAL_RUN_LAST,
236 G_STRUCT_OFFSET(GntFileSelClass, file_selected),
237 NULL, NULL,
238 gnt_closure_marshal_VOID__STRING_STRING,
239 G_TYPE_NONE, 0);
240 gnt_style_read_actions(G_OBJECT_CLASS_TYPE(klass), GNT_BINDABLE_CLASS(klass));
241
242 GNTDEBUG;
243 }
244
245 static void
246 gnt_file_sel_init(GTypeInstance *instance, gpointer class)
247 {
248 GNTDEBUG;
249 }
250
251 /******************************************************************************
252 * GntFileSel API
253 *****************************************************************************/
254 GType
255 gnt_file_sel_get_gtype(void)
256 {
257 static GType type = 0;
258
259 if(type == 0)
260 {
261 static const GTypeInfo info = {
262 sizeof(GntFileSelClass),
263 NULL, /* base_init */
264 NULL, /* base_finalize */
265 (GClassInitFunc)gnt_file_sel_class_init,
266 NULL, /* class_finalize */
267 NULL, /* class_data */
268 sizeof(GntFileSel),
269 0, /* n_preallocs */
270 gnt_file_sel_init, /* instance_init */
271 NULL
272 };
273
274 type = g_type_register_static(GNT_TYPE_WINDOW,
275 "GntFileSel",
276 &info, 0);
277 }
278
279 return type;
280 }
281
282 GntWidget *gnt_file_sel_new()
283 {
284 GntWidget *widget = g_object_new(GNT_TYPE_FILE_SEL, NULL);
285 GntFileSel *sel = GNT_FILE_SEL(widget);
286 GntWidget *hbox, *vbox;
287
288 vbox = gnt_vbox_new(FALSE);
289 gnt_box_set_pad(GNT_BOX(vbox), 0);
290 gnt_box_set_alignment(GNT_BOX(vbox), GNT_ALIGN_LEFT);
291
292 /* The dir. and files list */
293 hbox = gnt_hbox_new(FALSE);
294 gnt_box_set_pad(GNT_BOX(hbox), 0);
295
296 sel->dirs = gnt_tree_new();
297 gnt_tree_set_compare_func(GNT_TREE(sel->dirs), (GCompareFunc)g_utf8_collate);
298 gnt_tree_set_hash_fns(GNT_TREE(sel->dirs), g_str_hash, g_str_equal, g_free);
299 gnt_tree_set_column_titles(GNT_TREE(sel->dirs), "Directories");
300 gnt_tree_set_show_title(GNT_TREE(sel->dirs), TRUE);
301 gnt_tree_set_col_width(GNT_TREE(sel->dirs), 0, 20);
302 g_signal_connect(G_OBJECT(sel->dirs), "key_pressed", G_CALLBACK(dir_key_pressed), sel);
303
304 sel->files = gnt_tree_new_with_columns(2); /* Name, Size, Modified */
305 gnt_tree_set_compare_func(GNT_TREE(sel->files), (GCompareFunc)g_utf8_collate);
306 /*gnt_tree_set_column_titles(GNT_TREE(sel->files), "Filename", "Size", "Modified");*/
307 gnt_tree_set_column_titles(GNT_TREE(sel->files), "Filename", "Size");
308 gnt_tree_set_show_title(GNT_TREE(sel->files), TRUE);
309 gnt_tree_set_col_width(GNT_TREE(sel->files), 0, 25);
310 gnt_tree_set_col_width(GNT_TREE(sel->files), 1, 10);
311 /*gnt_tree_set_col_width(GNT_TREE(sel->files), 2, 10);*/
312 g_signal_connect(G_OBJECT(sel->files), "selection_changed", G_CALLBACK(file_sel_changed), sel);
313
314 gnt_box_add_widget(GNT_BOX(hbox), sel->dirs);
315 gnt_box_add_widget(GNT_BOX(hbox), sel->files);
316 gnt_box_add_widget(GNT_BOX(vbox), hbox);
317
318 /* The location entry */
319 sel->location = gnt_entry_new(NULL);
320 gnt_box_add_widget(GNT_BOX(vbox), sel->location);
321 g_signal_connect(G_OBJECT(sel->location), "key_pressed", G_CALLBACK(location_key_pressed), sel);
322
323 /* The buttons */
324 hbox = gnt_hbox_new(FALSE);
325 sel->cancel = gnt_button_new("Cancel");
326 sel->select = gnt_button_new("Select");
327 gnt_box_add_widget(GNT_BOX(hbox), sel->cancel);
328 gnt_box_add_widget(GNT_BOX(hbox), sel->select);
329 gnt_box_add_widget(GNT_BOX(vbox), hbox);
330
331 gnt_box_add_widget(GNT_BOX(sel), vbox);
332 return widget;
333 }
334
335 gboolean gnt_file_sel_set_current_location(GntFileSel *sel, const char *path)
336 {
337 char *old;
338 GError *error = NULL;
339 gboolean ret = TRUE;
340
341 old = sel->current;
342 sel->current = process_path(path);
343 if (!location_changed(sel, &error)) {
344 g_error_free(error);
345 error = NULL;
346 g_free(sel->current);
347 sel->current = old;
348 location_changed(sel, &error);
349 ret = FALSE;
350 } else
351 g_free(old);
352
353 update_location(sel);
354 return ret;
355 }
356