Parent Directory
|
Revision Log
Various fremantleization
1 | harbaum | 1 | /* |
2 | achadwick | 101 | * Copyright (C) 2008 Till Harbaum <till@harbaum.org>. |
3 | * | ||
4 | harbaum | 1 | * This file is part of OSM2Go. |
5 | * | ||
6 | * OSM2Go is free software: you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation, either version 3 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * OSM2Go is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License | ||
17 | * along with OSM2Go. If not, see <http://www.gnu.org/licenses/>. | ||
18 | */ | ||
19 | |||
20 | #include "appdata.h" | ||
21 | |||
22 | harbaum | 209 | static void vmessagef(GtkWidget *parent, int type, int buttons, |
23 | char *title, const char *fmt, | ||
24 | va_list args) { | ||
25 | |||
26 | harbaum | 1 | char *buf = g_strdup_vprintf(fmt, args); |
27 | |||
28 | harbaum | 315 | #if !defined(USE_HILDON) || (MAEMO_VERSION_MAJOR < 5) |
29 | harbaum | 1 | GtkWidget *dialog = gtk_message_dialog_new( |
30 | harbaum | 314 | GTK_WINDOW(parent), |
31 | GTK_DIALOG_DESTROY_WITH_PARENT, | ||
32 | type, buttons, buf); | ||
33 | harbaum | 1 | |
34 | gtk_window_set_title(GTK_WINDOW(dialog), title); | ||
35 | harbaum | 314 | #else |
36 | GtkWidget *dialog = | ||
37 | hildon_note_new_information(GTK_WINDOW(parent), buf); | ||
38 | #endif | ||
39 | harbaum | 1 | |
40 | gtk_dialog_run(GTK_DIALOG(dialog)); | ||
41 | gtk_widget_destroy(dialog); | ||
42 | |||
43 | g_free(buf); | ||
44 | } | ||
45 | |||
46 | harbaum | 209 | void messagef(GtkWidget *parent, char *title, const char *fmt, ...) { |
47 | va_list args; | ||
48 | va_start( args, fmt ); | ||
49 | harbaum | 314 | vmessagef(parent, GTK_MESSAGE_INFO, |
50 | harbaum | 294 | GTK_BUTTONS_OK, title, fmt, args); |
51 | harbaum | 209 | va_end( args ); |
52 | } | ||
53 | |||
54 | void errorf(GtkWidget *parent, const char *fmt, ...) { | ||
55 | va_list args; | ||
56 | va_start( args, fmt ); | ||
57 | harbaum | 294 | |
58 | harbaum | 314 | vmessagef(parent, GTK_MESSAGE_ERROR, |
59 | harbaum | 294 | GTK_BUTTONS_CLOSE, _("Error"), fmt, args); |
60 | harbaum | 209 | va_end( args ); |
61 | } | ||
62 | |||
63 | void warningf(GtkWidget *parent, const char *fmt, ...) { | ||
64 | va_list args; | ||
65 | va_start( args, fmt ); | ||
66 | harbaum | 314 | vmessagef(parent, GTK_MESSAGE_WARNING, |
67 | harbaum | 294 | GTK_BUTTONS_CLOSE, _("Warning"), fmt, args); |
68 | harbaum | 209 | va_end( args ); |
69 | } | ||
70 | |||
71 | harbaum | 315 | #ifndef FREMANTLE |
72 | #define RESPONSE_YES GTK_RESPONSE_YES | ||
73 | #define RESPONSE_NO GTK_RESPONSE_NO | ||
74 | #else | ||
75 | /* hildon names the yes/no buttons ok/cancel ??? */ | ||
76 | #define RESPONSE_YES GTK_RESPONSE_OK | ||
77 | #define RESPONSE_NO GTK_RESPONSE_CANCEL | ||
78 | #endif | ||
79 | |||
80 | harbaum | 1 | static void on_toggled(GtkWidget *button, gpointer data) { |
81 | harbaum | 315 | gboolean active = check_button_get_active(button); |
82 | harbaum | 1 | |
83 | GtkWidget *dialog = gtk_widget_get_toplevel(button); | ||
84 | |||
85 | if(*(gint*)data & MISC_AGAIN_FLAG_DONT_SAVE_NO) | ||
86 | gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog), | ||
87 | harbaum | 315 | RESPONSE_NO, !active); |
88 | harbaum | 1 | |
89 | if(*(gint*)data & MISC_AGAIN_FLAG_DONT_SAVE_YES) | ||
90 | gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog), | ||
91 | harbaum | 315 | RESPONSE_YES, !active); |
92 | harbaum | 1 | } |
93 | |||
94 | gboolean yes_no_f(GtkWidget *parent, appdata_t *appdata, gulong again_bit, | ||
95 | gint flags, char *title, const char *fmt, ...) { | ||
96 | |||
97 | if(appdata && again_bit && (appdata->dialog_again.not & again_bit)) | ||
98 | return((appdata->dialog_again.reply & again_bit) != 0); | ||
99 | |||
100 | va_list args; | ||
101 | va_start( args, fmt ); | ||
102 | char *buf = g_strdup_vprintf(fmt, args); | ||
103 | va_end( args ); | ||
104 | |||
105 | printf("%s: \"%s\"\n", title, buf); | ||
106 | |||
107 | harbaum | 315 | #ifndef FREMANTLE |
108 | harbaum | 1 | GtkWidget *dialog = gtk_message_dialog_new( |
109 | GTK_WINDOW(parent), | ||
110 | GTK_DIALOG_DESTROY_WITH_PARENT, | ||
111 | GTK_MESSAGE_QUESTION, GTK_BUTTONS_YES_NO, | ||
112 | buf); | ||
113 | |||
114 | gtk_window_set_title(GTK_WINDOW(dialog), title); | ||
115 | harbaum | 315 | #else |
116 | GtkWidget *dialog = | ||
117 | hildon_note_new_confirmation(GTK_WINDOW(parent), buf); | ||
118 | #endif | ||
119 | harbaum | 1 | |
120 | GtkWidget *cbut = NULL; | ||
121 | if(appdata && again_bit) { | ||
122 | harbaum | 315 | /* make sure there's some space before the checkbox */ |
123 | gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(dialog)->vbox), | ||
124 | gtk_label_new(" ")); | ||
125 | |||
126 | harbaum | 1 | GtkWidget *alignment = gtk_alignment_new(0.5, 0, 0, 0); |
127 | |||
128 | harbaum | 315 | cbut = check_button_new_with_label(_("Don't ask this question again")); |
129 | harbaum | 1 | g_signal_connect(cbut, "toggled", G_CALLBACK(on_toggled), &flags); |
130 | |||
131 | gtk_container_add(GTK_CONTAINER(alignment), cbut); | ||
132 | gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(dialog)->vbox), alignment); | ||
133 | |||
134 | gtk_widget_show_all(dialog); | ||
135 | } | ||
136 | |||
137 | harbaum | 315 | gboolean yes = (gtk_dialog_run(GTK_DIALOG(dialog)) == RESPONSE_YES); |
138 | harbaum | 1 | |
139 | if(cbut && gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(cbut))) { | ||
140 | /* the user doesn't want to see this dialog again */ | ||
141 | |||
142 | appdata->dialog_again.not |= again_bit; | ||
143 | if(yes) appdata->dialog_again.reply |= again_bit; | ||
144 | else appdata->dialog_again.reply &= ~again_bit; | ||
145 | } | ||
146 | |||
147 | gtk_widget_destroy(dialog); | ||
148 | |||
149 | g_free(buf); | ||
150 | return yes; | ||
151 | } | ||
152 | |||
153 | static const char *data_paths[] = { | ||
154 | harbaum | 41 | "~/." PACKAGE, // in home directory |
155 | harbaum | 308 | DATADIR , // final installation path |
156 | harbaum | 1 | #ifdef USE_HILDON |
157 | harbaum | 41 | "/media/mmc1/" PACKAGE, // path to external memory card |
158 | "/media/mmc2/" PACKAGE, // path to internal memory card | ||
159 | harbaum | 1 | #endif |
160 | harbaum | 41 | "./data", "../data", // local paths for testing |
161 | harbaum | 1 | NULL |
162 | }; | ||
163 | |||
164 | char *find_file(char *name) { | ||
165 | const char **path = data_paths; | ||
166 | char *p = getenv("HOME"); | ||
167 | |||
168 | while(*path) { | ||
169 | char *full_path = NULL; | ||
170 | |||
171 | if(*path[0] == '~') | ||
172 | full_path = g_strdup_printf("%s/%s/%s", p, *path+2, name); | ||
173 | else | ||
174 | full_path = g_strdup_printf("%s/%s", *path, name); | ||
175 | |||
176 | if(g_file_test(full_path, G_FILE_TEST_IS_REGULAR)) | ||
177 | return full_path; | ||
178 | |||
179 | g_free(full_path); | ||
180 | path++; | ||
181 | } | ||
182 | |||
183 | return NULL; | ||
184 | } | ||
185 | |||
186 | /* scan all data directories for the given file pattern and */ | ||
187 | /* return a list of files matching this pattern */ | ||
188 | file_chain_t *file_scan(char *pattern) { | ||
189 | file_chain_t *chain = NULL, **chainP = &chain; | ||
190 | |||
191 | const char **path = data_paths; | ||
192 | char *p = getenv("HOME"); | ||
193 | |||
194 | while(*path) { | ||
195 | GDir *dir = NULL; | ||
196 | |||
197 | /* scan for projects */ | ||
198 | const char *dirname = *path; | ||
199 | |||
200 | if(*path[0] == '~') | ||
201 | dirname = g_strdup_printf("%s/%s", p, *path+2); | ||
202 | |||
203 | if((dir = g_dir_open(dirname, 0, NULL))) { | ||
204 | const char *name = NULL; | ||
205 | do { | ||
206 | name = g_dir_read_name(dir); | ||
207 | |||
208 | if(name) { | ||
209 | char *fullname = g_strdup_printf("%s/%s", dirname, name); | ||
210 | if(g_file_test(fullname, G_FILE_TEST_IS_REGULAR)) { | ||
211 | if(g_pattern_match_simple(pattern, name)) { | ||
212 | *chainP = g_new0(file_chain_t, 1); | ||
213 | (*chainP)->name = fullname; | ||
214 | chainP = &(*chainP)->next; | ||
215 | } else | ||
216 | g_free(fullname); | ||
217 | } else | ||
218 | g_free(fullname); | ||
219 | } | ||
220 | } while(name); | ||
221 | |||
222 | g_dir_close(dir); | ||
223 | |||
224 | if(*path[0] == '~') | ||
225 | g_free((char*)dirname); | ||
226 | } | ||
227 | |||
228 | path++; | ||
229 | } | ||
230 | |||
231 | return chain; | ||
232 | } | ||
233 | harbaum | 167 | |
234 | |||
235 | #ifdef USE_HILDON | ||
236 | static const gint dialog_sizes[][2] = { | ||
237 | { 400, 100 }, // SMALL | ||
238 | harbaum | 266 | #if MAEMO_VERSION_MAJOR < 5 |
239 | harbaum | 167 | { 450, 300 }, // MEDIUM |
240 | { 800, 480 }, // LARGE | ||
241 | harbaum | 172 | #else |
242 | harbaum | 266 | /* in maemo5 most dialogs are full screen */ |
243 | { 800, 480 }, // MEDIUM | ||
244 | harbaum | 172 | { 800, 380 }, // LARGE |
245 | #endif | ||
246 | harbaum | 167 | { 640, 100 }, // WIDE |
247 | harbaum | 200 | { 450, 480 }, // HIGH |
248 | harbaum | 167 | }; |
249 | #else | ||
250 | static const gint dialog_sizes[][2] = { | ||
251 | { 300, 100 }, // SMALL | ||
252 | { 400, 300 }, // MEDIUM | ||
253 | { 500, 350 }, // LARGE | ||
254 | { 450, 100 }, // WIDE | ||
255 | harbaum | 200 | { 200, 350 }, // HIGH |
256 | harbaum | 167 | }; |
257 | #endif | ||
258 | |||
259 | /* create a modal dialog using one of the predefined size hints */ | ||
260 | GtkWidget *misc_dialog_new(guint hint, const char *title, | ||
261 | GtkWindow *parent, ...) { | ||
262 | va_list args; | ||
263 | va_start( args, parent ); | ||
264 | |||
265 | /* create dialog itself */ | ||
266 | GtkWidget *dialog = gtk_dialog_new(); | ||
267 | |||
268 | gtk_window_set_modal(GTK_WINDOW(dialog), TRUE); | ||
269 | if(title) gtk_window_set_title(GTK_WINDOW(dialog), title); | ||
270 | if(parent) gtk_window_set_transient_for(GTK_WINDOW(dialog), parent); | ||
271 | |||
272 | const gchar *button_text = va_arg(args, const gchar *); | ||
273 | while(button_text) { | ||
274 | gtk_dialog_add_button(GTK_DIALOG(dialog), button_text, va_arg(args, gint)); | ||
275 | button_text = va_arg(args, const gchar *); | ||
276 | } | ||
277 | |||
278 | va_end( args ); | ||
279 | |||
280 | if(hint != MISC_DIALOG_NOSIZE) | ||
281 | gtk_window_set_default_size(GTK_WINDOW(dialog), | ||
282 | dialog_sizes[hint][0], dialog_sizes[hint][1]); | ||
283 | |||
284 | return dialog; | ||
285 | } | ||
286 | harbaum | 195 | |
287 | #if defined(USE_HILDON) && (MAEMO_VERSION_MAJOR == 5) | ||
288 | #include <hildon/hildon-pannable-area.h> | ||
289 | /* create a pannable area */ | ||
290 | GtkWidget *misc_scrolled_window_new(gboolean etched_in) { | ||
291 | return hildon_pannable_area_new(); | ||
292 | } | ||
293 | |||
294 | void misc_scrolled_window_add_with_viewport(GtkWidget *win, GtkWidget *child) { | ||
295 | hildon_pannable_area_add_with_viewport(HILDON_PANNABLE_AREA(win), child); | ||
296 | } | ||
297 | |||
298 | #else | ||
299 | /* create a scrolled window */ | ||
300 | GtkWidget *misc_scrolled_window_new(gboolean etched_in) { | ||
301 | GtkWidget *scrolled_window = gtk_scrolled_window_new(NULL, NULL); | ||
302 | gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_window), | ||
303 | GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); | ||
304 | if(etched_in) | ||
305 | gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled_window), | ||
306 | GTK_SHADOW_ETCHED_IN); | ||
307 | return scrolled_window; | ||
308 | } | ||
309 | |||
310 | void misc_scrolled_window_add_with_viewport(GtkWidget *win, GtkWidget *child) { | ||
311 | gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(win), child); | ||
312 | } | ||
313 | |||
314 | |||
315 | #endif | ||
316 | |||
317 | harbaum | 200 | const char *misc_get_proxy_uri(settings_t *settings) { |
318 | static char proxy_buffer[64]; | ||
319 | harbaum | 195 | |
320 | harbaum | 200 | /* use environment settings if preset */ |
321 | const char *proxy = g_getenv("http_proxy"); | ||
322 | harbaum | 232 | if(proxy) { |
323 | printf("http_proxy: %s\n", proxy); | ||
324 | return proxy; | ||
325 | } | ||
326 | harbaum | 200 | |
327 | /* otherwise try settings */ | ||
328 | if(!settings || !settings->proxy || | ||
329 | !settings->proxy->host) return NULL; | ||
330 | |||
331 | harbaum | 232 | snprintf(proxy_buffer, sizeof(proxy_buffer), "%s%s:%u", |
332 | strncmp(settings->proxy->host, "http://", 7)?"http://":"", | ||
333 | settings->proxy->host, settings->proxy->port); | ||
334 | |||
335 | harbaum | 200 | proxy_buffer[sizeof(proxy_buffer)-1] = 0; |
336 | harbaum | 232 | printf("gconf_proxy: %s\n", proxy_buffer); |
337 | harbaum | 200 | return proxy_buffer; |
338 | } | ||
339 | harbaum | 203 | |
340 | void misc_table_attach(GtkWidget *table, GtkWidget *widget, int x, int y) { | ||
341 | gtk_table_attach_defaults(GTK_TABLE(table), widget, x, x+1, y, y+1); | ||
342 | } | ||
343 | harbaum | 315 | |
344 | /* ---------- unified widgets for fremantle/others --------------- */ | ||
345 | |||
346 | GtkWidget *entry_new(void) { | ||
347 | #ifndef FREMANTLE | ||
348 | return gtk_entry_new(); | ||
349 | #else | ||
350 | return hildon_entry_new(HILDON_SIZE_AUTO); | ||
351 | #endif | ||
352 | } | ||
353 | |||
354 | GtkWidget *button_new(void) { | ||
355 | GtkWidget *button = gtk_button_new(); | ||
356 | #ifdef FREMANTLE | ||
357 | hildon_gtk_widget_set_theme_size(button, | ||
358 | (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH)); | ||
359 | #endif | ||
360 | return button; | ||
361 | } | ||
362 | |||
363 | GtkWidget *button_new_with_label(char *label) { | ||
364 | GtkWidget *button = gtk_button_new_with_label(label); | ||
365 | #ifdef FREMANTLE | ||
366 | hildon_gtk_widget_set_theme_size(button, | ||
367 | (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH)); | ||
368 | #endif | ||
369 | return button; | ||
370 | } | ||
371 | |||
372 | GtkWidget *check_button_new_with_label(char *label) { | ||
373 | #ifndef FREMANTLE | ||
374 | return gtk_check_button_new_with_label(label); | ||
375 | #else | ||
376 | GtkWidget *cbut = | ||
377 | hildon_check_button_new(HILDON_SIZE_FINGER_HEIGHT | | ||
378 | HILDON_SIZE_AUTO_WIDTH); | ||
379 | gtk_button_set_label(GTK_BUTTON(cbut), label); | ||
380 | return cbut; | ||
381 | #endif | ||
382 | } | ||
383 | |||
384 | void check_button_set_active(GtkWidget *button, gboolean active) { | ||
385 | #ifndef FREMANTLE | ||
386 | gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), active); | ||
387 | #else | ||
388 | hildon_check_button_set_active(HILDON_CHECK_BUTTON(button), active); | ||
389 | #endif | ||
390 | } | ||
391 | |||
392 | gboolean check_button_get_active(GtkWidget *button) { | ||
393 | #ifndef FREMANTLE | ||
394 | return gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button)); | ||
395 | #else | ||
396 | return hildon_check_button_get_active(HILDON_CHECK_BUTTON(button)); | ||
397 | #endif | ||
398 | } | ||
399 | |||
400 | GtkWidget *notebook_new(void) { | ||
401 | #ifdef FREMANTLE | ||
402 | GtkWidget *vbox = gtk_vbox_new(FALSE, 0); | ||
403 | |||
404 | GtkWidget *notebook = gtk_notebook_new(); | ||
405 | |||
406 | /* solution for fremantle: we use a row of ordinary buttons instead */ | ||
407 | /* of regular tabs */ | ||
408 | |||
409 | /* hide the regular tabs */ | ||
410 | gtk_notebook_set_show_tabs(GTK_NOTEBOOK(notebook), FALSE); | ||
411 | |||
412 | gtk_box_pack_start_defaults(GTK_BOX(vbox), notebook); | ||
413 | |||
414 | /* store a reference to the notebook in the vbox */ | ||
415 | g_object_set_data(G_OBJECT(vbox), "notebook", (gpointer)notebook); | ||
416 | |||
417 | /* create a hbox for the buttons */ | ||
418 | GtkWidget *hbox = gtk_hbox_new(TRUE, 0); | ||
419 | gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); | ||
420 | g_object_set_data(G_OBJECT(vbox), "hbox", (gpointer)hbox); | ||
421 | |||
422 | return vbox; | ||
423 | #else | ||
424 | return gtk_notebook_new(); | ||
425 | #endif | ||
426 | } | ||
427 | |||
428 | GtkWidget *notebook_get_gtk_notebook(GtkWidget *notebook) { | ||
429 | #ifdef FREMANTLE | ||
430 | return GTK_WIDGET(g_object_get_data(G_OBJECT(notebook), "notebook")); | ||
431 | #else | ||
432 | return notebook; | ||
433 | #endif | ||
434 | } | ||
435 | |||
436 | |||
437 | #ifdef FREMANTLE | ||
438 | static void on_notebook_button_clicked(GtkWidget *button, gpointer data) { | ||
439 | GtkNotebook *nb = | ||
440 | GTK_NOTEBOOK(g_object_get_data(G_OBJECT(data), "notebook")); | ||
441 | |||
442 | gint page = (gint)g_object_get_data(G_OBJECT(button), "page"); | ||
443 | gtk_notebook_set_current_page(nb, page); | ||
444 | } | ||
445 | #endif | ||
446 | |||
447 | void notebook_append_page(GtkWidget *notebook, | ||
448 | GtkWidget *page, char *label) { | ||
449 | #ifdef FREMANTLE | ||
450 | GtkNotebook *nb = | ||
451 | GTK_NOTEBOOK(g_object_get_data(G_OBJECT(notebook), "notebook")); | ||
452 | |||
453 | gint page_num = gtk_notebook_append_page(nb, page, gtk_label_new(label)); | ||
454 | GtkWidget *button = NULL; | ||
455 | |||
456 | /* select button for page 0 by default */ | ||
457 | if(!page_num) { | ||
458 | button = gtk_radio_button_new_with_label(NULL, label); | ||
459 | gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), TRUE); | ||
460 | g_object_set_data(G_OBJECT(notebook), "group_master", (gpointer)button); | ||
461 | } else { | ||
462 | GtkWidget *master = g_object_get_data(G_OBJECT(notebook), "group_master"); | ||
463 | button = gtk_radio_button_new_with_label_from_widget( | ||
464 | GTK_RADIO_BUTTON(master), label); | ||
465 | } | ||
466 | |||
467 | gtk_toggle_button_set_mode(GTK_TOGGLE_BUTTON(button), FALSE); | ||
468 | g_object_set_data(G_OBJECT(button), "page", (gpointer)page_num); | ||
469 | |||
470 | gtk_signal_connect(GTK_OBJECT(button), "clicked", | ||
471 | GTK_SIGNAL_FUNC(on_notebook_button_clicked), notebook); | ||
472 | |||
473 | #if defined(USE_HILDON) && (MAEMO_VERSION_MAJOR == 5) | ||
474 | hildon_gtk_widget_set_theme_size(button, | ||
475 | (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH)); | ||
476 | #endif | ||
477 | |||
478 | gtk_box_pack_start_defaults( | ||
479 | GTK_BOX(g_object_get_data(G_OBJECT(notebook), "hbox")), | ||
480 | button); | ||
481 | |||
482 | #else | ||
483 | gtk_notebook_append_page(GTK_NOTEBOOK(notebook), page, gtk_label_new(label)); | ||
484 | #endif | ||
485 | } | ||
486 |