]> git.street.me.uk Git - andy/viking.git/blob - src/viklayerspanel.c
Be more selective in collecting realtime trackpoints.
[andy/viking.git] / src / viklayerspanel.c
1 /*
2  * viking -- GPS Data and Topo Analyzer, Explorer, and Manager
3  *
4  * Copyright (C) 2003-2005, Evan Battaglia <gtoevan@gmx.net>
5  *
6  * This program 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 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program 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 this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  */
21
22 #include "viking.h"
23
24 #include <string.h>
25
26 enum {
27   VLP_UPDATE_SIGNAL,
28   VLP_LAST_SIGNAL
29 };
30
31 static void layers_panel_class_init ( VikLayersPanelClass *klass );
32 static void layers_panel_init ( VikLayersPanel *vlp );
33 static void layers_item_edited (VikLayersPanel *vlp, GtkTreeIter *iter, const gchar *new_text);
34 static void layers_item_toggled (VikLayersPanel *vlp, GtkTreeIter *iter);
35
36 static guint layers_panel_signals[VLP_LAST_SIGNAL] = { 0 };
37
38 static GObjectClass *parent_class;
39
40 struct _VikLayersPanel {
41   GtkVBox vbox;
42
43   VikAggregateLayer *toplayer;
44   GtkTreeIter toplayer_iter;
45
46   VikTreeview *vt;
47   VikViewport *vvp; /* reference */
48
49   GtkItemFactory *popup_factory;
50 };
51
52 static GtkItemFactoryEntry base_entries[] = {
53  { "/C_ut", NULL, (GtkItemFactoryCallback) vik_layers_panel_cut_selected, -1, "<StockItem>", GTK_STOCK_CUT },
54  { "/_Copy", NULL, (GtkItemFactoryCallback) vik_layers_panel_copy_selected, -1, "<StockItem>", GTK_STOCK_COPY },
55  { "/_Paste", NULL, (GtkItemFactoryCallback) vik_layers_panel_paste_selected, -1, "<StockItem>", GTK_STOCK_PASTE },
56  { "/_Delete", NULL, (GtkItemFactoryCallback) vik_layers_panel_delete_selected, -1, "<StockItem>", GTK_STOCK_DELETE },
57  { "/New Layer", NULL, NULL, -1, "<Branch>" },
58 };
59
60 #define NUM_BASE_ENTRIES 5
61
62 static void layers_item_toggled (VikLayersPanel *vlp, GtkTreeIter *iter);
63 static void layers_item_edited (VikLayersPanel *vlp, GtkTreeIter *iter, const gchar *new_text);
64 static void layers_popup_cb (VikLayersPanel *vlp);
65 static void layers_popup ( VikLayersPanel *vlp, GtkTreeIter *iter, gint mouse_button );
66 static gboolean layers_button_press_cb (VikLayersPanel *vlp, GdkEventButton *event);
67 static void layers_move_item ( VikLayersPanel *vlp, gboolean up );
68 static void layers_move_item_up ( VikLayersPanel *vlp );
69 static void layers_move_item_down ( VikLayersPanel *vlp );
70 static void layers_panel_finalize ( GObject *gob );
71
72 GType vik_layers_panel_get_type()
73 {
74   static GType vlp_type = 0;
75
76   if (!vlp_type)
77   {
78     static const GTypeInfo vlp_info = 
79     {
80       sizeof (VikLayersPanelClass),
81       NULL, /* base_init */
82       NULL, /* base_finalize */
83       (GClassInitFunc) layers_panel_class_init,
84       NULL, /* class_finalize */
85       NULL, /* class_data */
86       sizeof (VikLayersPanel),
87       0,
88       (GInstanceInitFunc) layers_panel_init,
89     };
90     vlp_type = g_type_register_static ( GTK_TYPE_VBOX, "VikLayersPanel", &vlp_info, 0 );
91   }
92
93   return vlp_type;
94 }
95
96 static void layers_panel_class_init ( VikLayersPanelClass *klass )
97 {
98   GObjectClass *object_class;
99
100   object_class = G_OBJECT_CLASS (klass);
101
102   object_class->finalize = layers_panel_finalize;
103
104   parent_class = g_type_class_peek_parent (klass);
105
106   layers_panel_signals[VLP_UPDATE_SIGNAL] = g_signal_new ( "update", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (VikLayersPanelClass, update), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
107 }
108
109 VikLayersPanel *vik_layers_panel_new ()
110 {
111   return VIK_LAYERS_PANEL ( g_object_new ( VIK_LAYERS_PANEL_TYPE, NULL ) );
112 }
113
114 void vik_layers_panel_set_viewport ( VikLayersPanel *vlp, VikViewport *vvp )
115 {
116   vlp->vvp = vvp;
117   /* TODO: also update GCs (?) */
118 }
119
120 VikViewport *vik_layers_panel_get_viewport ( VikLayersPanel *vlp )
121 {
122   return vlp->vvp;
123 }
124
125 static void layers_panel_init ( VikLayersPanel *vlp )
126 {
127   GtkWidget *hbox;
128   GtkWidget *upbutton, *upimage, *downbutton, *downimage;
129   GtkWidget *scrolledwindow;
130   GtkItemFactoryEntry entry;
131   guint i, tmp;
132
133   vlp->vvp = NULL;
134
135   hbox = gtk_hbox_new ( TRUE, 2 );
136   vlp->vt = vik_treeview_new ( );
137
138   vlp->toplayer = vik_aggregate_layer_new ();
139   vik_layer_rename ( VIK_LAYER(vlp->toplayer), "Top Layer");
140   g_signal_connect_swapped ( G_OBJECT(vlp->toplayer), "update", G_CALLBACK(vik_layers_panel_emit_update), vlp );
141
142   vik_treeview_add_layer ( vlp->vt, NULL, &(vlp->toplayer_iter), VIK_LAYER(vlp->toplayer)->name, NULL, vlp->toplayer, VIK_LAYER_AGGREGATE, VIK_LAYER_AGGREGATE );
143   vik_layer_realize ( VIK_LAYER(vlp->toplayer), vlp->vt, &(vlp->toplayer_iter) );
144
145   g_signal_connect_swapped ( vlp->vt, "popup_menu", G_CALLBACK(layers_popup_cb), vlp);
146   g_signal_connect_swapped ( vlp->vt, "button_press_event", G_CALLBACK(layers_button_press_cb), vlp);
147   g_signal_connect_swapped ( vlp->vt, "item_toggled", G_CALLBACK(layers_item_toggled), vlp);
148   g_signal_connect_swapped ( vlp->vt, "item_edited", G_CALLBACK(layers_item_edited), vlp);
149
150   upimage = gtk_image_new_from_stock ( GTK_STOCK_GO_UP, GTK_ICON_SIZE_BUTTON );
151   upbutton = gtk_button_new ( );
152   gtk_container_add ( GTK_CONTAINER(upbutton), upimage );
153   gtk_box_pack_start ( GTK_BOX(hbox), upbutton, TRUE, TRUE, 0 );
154   g_signal_connect_swapped ( G_OBJECT(upbutton), "clicked", G_CALLBACK(layers_move_item_up), vlp );
155   downimage = gtk_image_new_from_stock ( GTK_STOCK_GO_DOWN, GTK_ICON_SIZE_BUTTON );
156   downbutton = gtk_button_new ( );
157   gtk_container_add ( GTK_CONTAINER(downbutton), downimage );
158   gtk_box_pack_start ( GTK_BOX(hbox), downbutton, TRUE, TRUE, 0 );
159   g_signal_connect_swapped ( G_OBJECT(downbutton), "clicked", G_CALLBACK(layers_move_item_down), vlp );
160
161   scrolledwindow = gtk_scrolled_window_new ( NULL, NULL );
162   gtk_scrolled_window_set_policy ( GTK_SCROLLED_WINDOW(scrolledwindow), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC );
163   gtk_container_add ( GTK_CONTAINER(scrolledwindow), GTK_WIDGET(vlp->vt) );
164   
165   gtk_box_pack_start ( GTK_BOX(vlp), scrolledwindow, TRUE, TRUE, 0 );
166   gtk_box_pack_start ( GTK_BOX(vlp), hbox, FALSE, FALSE, 0 );
167
168   vlp->popup_factory = gtk_item_factory_new ( GTK_TYPE_MENU, "<main>", NULL );
169   gtk_item_factory_create_items ( vlp->popup_factory, NUM_BASE_ENTRIES, base_entries, vlp );
170   for ( i = 0; i < VIK_LAYER_NUM_TYPES; i++ )
171   {
172     /* TODO: FIXME: if name has a '/' in it it will get all messed up. why not have an itemfactory field with
173                     name, icon, shortcut, etc.? */
174     entry.path = g_strdup_printf("%s/New %s Layer", base_entries[NUM_BASE_ENTRIES-1].path, vik_layer_get_interface(i)->name );
175     entry.accelerator = NULL;
176     entry.callback = (GtkItemFactoryCallback) vik_layers_panel_new_layer;
177     entry.callback_action = i;
178     if ( vik_layer_get_interface(i)->icon )
179     {
180       entry.item_type = "<ImageItem>";
181       entry.extra_data = gdk_pixdata_serialize ( vik_layer_get_interface(i)->icon, &tmp );
182     }
183     else
184       entry.item_type = "<Item>";
185
186     gtk_item_factory_create_item ( vlp->popup_factory, &entry, vlp, 1 );
187     g_free ( (gpointer) entry.extra_data );
188     g_free ( entry.path );
189   }
190 }
191
192 void vik_layers_panel_emit_update ( VikLayersPanel *vlp )
193 {
194   g_signal_emit ( G_OBJECT(vlp), layers_panel_signals[VLP_UPDATE_SIGNAL], 0 );
195 }
196
197 static void layers_item_toggled (VikLayersPanel *vlp, GtkTreeIter *iter)
198 {
199   gboolean visible;
200   gpointer p;
201   gint type;
202
203   /* get type and data */
204   type = vik_treeview_item_get_type ( vlp->vt, iter );
205   p = vik_treeview_item_get_pointer ( vlp->vt, iter );
206
207   switch ( type )
208   {
209     case VIK_TREEVIEW_TYPE_LAYER:
210       visible = (VIK_LAYER(p)->visible ^= 1);
211       vik_layer_emit_update_although_invisible ( VIK_LAYER(p) ); /* set trigger for half-drawn */
212       break;
213     case VIK_TREEVIEW_TYPE_SUBLAYER:
214       visible = vik_layer_sublayer_toggle_visible ( VIK_LAYER(vik_treeview_item_get_parent(vlp->vt, iter)),
215                                                 vik_treeview_item_get_data(vlp->vt, iter), p);
216       vik_layer_emit_update_although_invisible ( VIK_LAYER(vik_treeview_item_get_parent(vlp->vt, iter)) );
217       break;
218     default: return;
219   }
220
221   vik_treeview_item_set_visible ( vlp->vt, iter, visible );
222 }
223
224 static void layers_item_edited (VikLayersPanel *vlp, GtkTreeIter *iter, const gchar *new_text)
225 {
226   if ( vik_treeview_item_get_type ( vlp->vt, iter ) == VIK_TREEVIEW_TYPE_LAYER )
227   {
228     VikLayer *l;
229
230     /* get iter and layer */
231     l = VIK_LAYER ( vik_treeview_item_get_pointer ( vlp->vt, iter ) );
232
233     if ( strcmp ( l->name, new_text ) != 0 )
234     {
235       vik_layer_rename ( l, new_text );
236       vik_treeview_item_set_name ( vlp->vt, iter, l->name );
237     }
238   }
239   else
240   {
241     const gchar *name = vik_layer_sublayer_rename_request ( vik_treeview_item_get_parent ( vlp->vt, iter ), new_text, vlp, vik_treeview_item_get_data ( vlp->vt, iter ), vik_treeview_item_get_pointer ( vlp->vt, iter ), iter );
242     if ( name )
243       vik_treeview_item_set_name ( vlp->vt, iter, name);
244   }
245 }
246
247 static gboolean layers_button_press_cb ( VikLayersPanel *vlp, GdkEventButton *event )
248 {
249   if (event->button == 3)
250   {
251     GtkTreeIter iter;
252     if ( vik_treeview_get_iter_at_pos ( vlp->vt, &iter, event->x, event->y ) )
253     {
254       layers_popup ( vlp, &iter, 3 );
255       vik_treeview_item_select ( vlp->vt, &iter );
256     }
257     else
258       layers_popup ( vlp, NULL, 3 );
259     return TRUE;
260   }
261   return FALSE;
262 }
263
264 static void layers_popup ( VikLayersPanel *vlp, GtkTreeIter *iter, gint mouse_button )
265 {
266   GtkMenu *menu = NULL;
267
268
269   if ( iter )
270   {
271     if ( vik_treeview_item_get_type ( vlp->vt, iter ) == VIK_TREEVIEW_TYPE_LAYER )
272     {
273       VikLayer *layer = VIK_LAYER(vik_treeview_item_get_pointer ( vlp->vt, iter ));
274
275       if ( layer->type == VIK_LAYER_AGGREGATE )
276         menu = GTK_MENU(gtk_item_factory_get_widget ( vlp->popup_factory, "<main>" ));
277       else
278       {
279         GtkWidget *del, *prop;
280         VikStdLayerMenuItem menu_selection = vik_layer_get_menu_items_selection(layer);
281
282         menu = GTK_MENU ( gtk_menu_new () );
283
284         if (menu_selection & VIK_MENU_ITEM_PROPERTY) {
285           prop = gtk_image_menu_item_new_from_stock ( GTK_STOCK_PROPERTIES, NULL );
286           g_signal_connect_swapped ( G_OBJECT(prop), "activate", G_CALLBACK(vik_layers_panel_properties), vlp );
287           gtk_menu_shell_append (GTK_MENU_SHELL (menu), prop);
288           gtk_widget_show ( prop );
289         }
290
291         if (menu_selection & VIK_MENU_ITEM_CUT) {
292           del = gtk_image_menu_item_new_from_stock ( GTK_STOCK_CUT, NULL );
293           g_signal_connect_swapped ( G_OBJECT(del), "activate", G_CALLBACK(vik_layers_panel_cut_selected), vlp );
294           gtk_menu_shell_append (GTK_MENU_SHELL (menu), del);
295           gtk_widget_show ( del );
296         }
297
298         if (menu_selection & VIK_MENU_ITEM_COPY) {
299           del = gtk_image_menu_item_new_from_stock ( GTK_STOCK_COPY, NULL );
300           g_signal_connect_swapped ( G_OBJECT(del), "activate", G_CALLBACK(vik_layers_panel_copy_selected), vlp );
301           gtk_menu_shell_append (GTK_MENU_SHELL (menu), del);
302           gtk_widget_show ( del );
303         }
304
305         if (menu_selection & VIK_MENU_ITEM_PASTE) {
306           del = gtk_image_menu_item_new_from_stock ( GTK_STOCK_PASTE, NULL );
307           g_signal_connect_swapped ( G_OBJECT(del), "activate", G_CALLBACK(vik_layers_panel_paste_selected), vlp );
308           gtk_menu_shell_append (GTK_MENU_SHELL (menu), del);
309           gtk_widget_show ( del );
310         }
311
312         if (menu_selection & VIK_MENU_ITEM_DELETE) {
313           del = gtk_image_menu_item_new_from_stock ( GTK_STOCK_DELETE, NULL );
314           g_signal_connect_swapped ( G_OBJECT(del), "activate", G_CALLBACK(vik_layers_panel_delete_selected), vlp );
315           gtk_menu_shell_append (GTK_MENU_SHELL (menu), del);
316           gtk_widget_show ( del );
317         }
318
319         vik_layer_add_menu_items ( layer, menu, vlp );
320       } 
321     }
322     else
323     {
324       menu = GTK_MENU ( gtk_menu_new () );
325       if ( ! vik_layer_sublayer_add_menu_items ( vik_treeview_item_get_parent ( vlp->vt, iter ), menu, vlp, vik_treeview_item_get_data ( vlp->vt, iter ), vik_treeview_item_get_pointer ( vlp->vt, iter ), iter ) )
326       {
327         gtk_widget_destroy ( GTK_WIDGET(menu) );
328         return;
329       }
330       /* TODO: specific things for different types */
331     }
332   }
333   else
334     menu = GTK_MENU(gtk_item_factory_get_widget ( vlp->popup_factory, base_entries[NUM_BASE_ENTRIES-1].path ));
335   gtk_menu_popup ( menu, NULL, NULL, NULL, NULL, mouse_button, gtk_get_current_event_time() );
336 }
337
338 static void layers_popup_cb ( VikLayersPanel *vlp )
339 {
340   GtkTreeIter iter;
341   layers_popup ( vlp, vik_treeview_get_selected_iter ( vlp->vt, &iter ) ? &iter : NULL, 0 );
342 }
343
344 gboolean vik_layers_panel_new_layer ( VikLayersPanel *vlp, gint type )
345 {
346   VikLayer *l;
347   g_assert ( vlp->vvp );
348   l = vik_layer_create ( type, vlp->vvp, VIK_GTK_WINDOW_FROM_WIDGET(vlp), TRUE );
349   if ( l )
350   {
351     vik_layers_panel_add_layer ( vlp, l );
352     vik_layers_panel_emit_update ( vlp );
353     return TRUE;
354   }
355   return FALSE;
356 }
357
358 void vik_layers_panel_add_layer ( VikLayersPanel *vlp, VikLayer *l )
359 {
360   GtkTreeIter iter;
361   GtkTreeIter *replace_iter = NULL;
362
363   /* could be something different so we have to do this */
364   vik_layer_change_coord_mode ( l, vik_viewport_get_coord_mode(vlp->vvp) );
365
366   if ( ! vik_treeview_get_selected_iter ( vlp->vt, &iter ) )
367     vik_aggregate_layer_add_layer ( vlp->toplayer, l );
368   else
369   {
370     VikAggregateLayer *addtoagg;
371     if (vik_treeview_item_get_type ( vlp->vt, &iter ) == VIK_TREEVIEW_TYPE_LAYER )
372     {
373       if ( IS_VIK_AGGREGATE_LAYER(vik_treeview_item_get_pointer ( vlp->vt, &iter )) )
374          addtoagg = VIK_AGGREGATE_LAYER(vik_treeview_item_get_pointer ( vlp->vt, &iter ));
375       else {
376        VikLayer *vl = VIK_LAYER(vik_treeview_item_get_parent ( vlp->vt, &iter ));
377        while ( ! IS_VIK_AGGREGATE_LAYER(vl) ) {
378          iter = vl->iter;
379          vl = VIK_LAYER(vik_treeview_item_get_parent ( vlp->vt, &vl->iter ));
380          g_assert ( vl->realized );
381        }
382        addtoagg = VIK_AGGREGATE_LAYER(vl);
383        replace_iter = &iter;
384       }
385     }
386     else
387     {
388       /* a sublayer is selected, first get its parent (layer), then find the layer's parent (aggr. layer) */
389       VikLayer *vl = VIK_LAYER(vik_treeview_item_get_parent ( vlp->vt, &iter ));
390       replace_iter = &(vl->iter);
391       g_assert ( vl->realized );
392       VikLayer *grandpa = (vik_treeview_item_get_parent ( vlp->vt, &(vl->iter) ) );
393       if (IS_VIK_AGGREGATE_LAYER(grandpa))
394         addtoagg = VIK_AGGREGATE_LAYER(grandpa);
395       else {
396         addtoagg = vlp->toplayer;
397         replace_iter = &grandpa->iter;
398       }
399     }
400     if ( replace_iter )
401       vik_aggregate_layer_insert_layer ( addtoagg, l, replace_iter );
402     else
403       vik_aggregate_layer_add_layer ( addtoagg, l );
404   }
405 }
406
407 static void layers_move_item ( VikLayersPanel *vlp, gboolean up )
408 {
409   GtkTreeIter iter;
410   VikAggregateLayer *parent;
411
412   /* TODO: deactivate the buttons and stuff */
413   if ( ! vik_treeview_get_selected_iter ( vlp->vt, &iter ) )
414     return;
415
416   vik_treeview_select_iter ( vlp->vt, &iter ); /* cancel any layer-name editing going on... */
417
418   if ( vik_treeview_item_get_type ( vlp->vt, &iter ) == VIK_TREEVIEW_TYPE_LAYER )
419   {
420     parent = VIK_AGGREGATE_LAYER(vik_treeview_item_get_parent ( vlp->vt, &iter ));
421     if ( parent ) /* not toplevel */
422     {
423       vik_aggregate_layer_move_layer ( parent, &iter, up );
424       vik_layers_panel_emit_update ( vlp );
425     }
426   }
427 }
428
429 gboolean vik_layers_panel_properties ( VikLayersPanel *vlp )
430 {
431   GtkTreeIter iter;
432   g_assert ( vlp->vvp );
433
434   if ( vik_treeview_get_selected_iter ( vlp->vt, &iter ) && vik_treeview_item_get_type ( vlp->vt, &iter ) == VIK_TREEVIEW_TYPE_LAYER )
435   {
436     if ( vik_treeview_item_get_data ( vlp->vt, &iter ) == VIK_LAYER_AGGREGATE )
437       a_dialog_info_msg ( VIK_GTK_WINDOW_FROM_WIDGET(vlp), "Aggregate Layers have no settable properties." );
438     vik_layer_properties ( VIK_LAYER( vik_treeview_item_get_pointer ( vlp->vt, &iter ) ), vlp->vvp );
439     return TRUE;
440   }
441   else
442     return FALSE;
443 }
444
445 void vik_layers_panel_draw_all ( VikLayersPanel *vlp )
446 {
447   if ( vlp->vvp && VIK_LAYER(vlp->toplayer)->visible )
448     vik_aggregate_layer_draw ( vlp->toplayer, vlp->vvp );
449 }
450
451 void vik_layers_panel_draw_all_using_viewport ( VikLayersPanel *vlp, VikViewport *vvp )
452 {
453   if ( vlp->vvp && VIK_LAYER(vlp->toplayer)->visible )
454     vik_aggregate_layer_draw ( vlp->toplayer, vvp );
455 }
456
457 void vik_layers_panel_cut_selected ( VikLayersPanel *vlp )
458 {
459   gint type;
460   GtkTreeIter iter;
461   
462   g_return_if_fail ( vik_treeview_get_selected_iter ( vlp->vt, &iter ) );
463
464   type = vik_treeview_item_get_type ( vlp->vt, &iter );
465
466   if ( type == VIK_TREEVIEW_TYPE_LAYER )
467   {
468     VikAggregateLayer *parent = vik_treeview_item_get_parent ( vlp->vt, &iter );
469
470     if ( parent )
471     {
472
473       /* reset trigger if trigger deleted */
474       if ( vik_layers_panel_get_selected ( vlp ) == vik_viewport_get_trigger ( vlp->vvp ) )
475         vik_viewport_set_trigger ( vlp->vvp, NULL );
476
477       a_clipboard_copy_selected ( vlp );
478       if ( vik_aggregate_layer_delete ( parent, &iter ) )
479         vik_layers_panel_emit_update ( vlp );
480     }
481     else
482       a_dialog_info_msg ( VIK_GTK_WINDOW_FROM_WIDGET(vlp), "You cannot cut the Top Layer." );
483   }
484 }
485
486 void vik_layers_panel_copy_selected ( VikLayersPanel *vlp )
487 {
488   gint type;
489   GtkTreeIter iter;
490   
491   g_return_if_fail ( vik_treeview_get_selected_iter ( vlp->vt, &iter ) );
492
493   type = vik_treeview_item_get_type ( vlp->vt, &iter );
494
495   if ( type == VIK_TREEVIEW_TYPE_LAYER ) {
496     a_clipboard_copy_selected ( vlp );
497   }
498 }
499
500 void vik_layers_panel_paste_selected ( VikLayersPanel *vlp )
501 {
502   GtkTreeIter iter;
503   g_return_if_fail ( vik_treeview_get_selected_iter ( vlp->vt, &iter ) );
504   a_clipboard_paste ( vlp );
505 }
506
507 void vik_layers_panel_delete_selected ( VikLayersPanel *vlp )
508 {
509   gint type;
510   GtkTreeIter iter;
511   
512   g_return_if_fail ( vik_treeview_get_selected_iter ( vlp->vt, &iter ) );
513
514   type = vik_treeview_item_get_type ( vlp->vt, &iter );
515
516   if ( type == VIK_TREEVIEW_TYPE_LAYER )
517   {
518     VikAggregateLayer *parent = vik_treeview_item_get_parent ( vlp->vt, &iter );
519     if ( parent )
520     {
521       /* reset trigger if trigger deleted */
522       if ( vik_layers_panel_get_selected ( vlp ) == vik_viewport_get_trigger ( vlp->vvp ) )
523         vik_viewport_set_trigger ( vlp->vvp, NULL );
524
525       if (IS_VIK_AGGREGATE_LAYER(parent)) {
526         if ( vik_aggregate_layer_delete ( parent, &iter ) )
527           vik_layers_panel_emit_update ( vlp );
528       }
529     }
530     else
531       a_dialog_info_msg ( VIK_GTK_WINDOW_FROM_WIDGET(vlp), "You cannot delete the Top Layer." );
532   }
533   else if (type == VIK_TREEVIEW_TYPE_SUBLAYER) {
534     VikLayer *sel = vik_layers_panel_get_selected ( vlp );
535     if ( vik_layer_get_interface(sel->type)->delete_item ) {
536       gint subtype = vik_treeview_item_get_data( vlp->vt, &iter);
537       vik_layer_get_interface(sel->type)->delete_item ( sel, subtype, vik_treeview_item_get_pointer(sel->vt, &iter) );
538     }
539   }
540 }
541
542 VikLayer *vik_layers_panel_get_selected ( VikLayersPanel *vlp )
543 {
544   GtkTreeIter iter, parent;
545   gint type;
546
547   if ( ! vik_treeview_get_selected_iter ( vlp->vt, &iter ) )
548     return NULL;
549
550   type = vik_treeview_item_get_type ( vlp->vt, &iter );
551
552   while ( type != VIK_TREEVIEW_TYPE_LAYER )
553   {
554     if ( ! vik_treeview_item_get_parent_iter ( vlp->vt, &iter, &parent ) )
555       return NULL;
556     iter = parent;
557     type = vik_treeview_item_get_type ( vlp->vt, &iter );
558   }
559
560   return VIK_LAYER( vik_treeview_item_get_pointer ( vlp->vt, &iter ) );
561 }
562
563 static void layers_move_item_up ( VikLayersPanel *vlp )
564 {
565   layers_move_item ( vlp, TRUE );
566 }
567
568 static void layers_move_item_down ( VikLayersPanel *vlp )
569 {
570   layers_move_item ( vlp, FALSE );
571 }
572
573 #if 0
574 gboolean vik_layers_panel_tool ( VikLayersPanel *vlp, guint16 layer_type, VikToolInterfaceFunc tool_func, GdkEventButton *event, VikViewport *vvp )
575 {
576   VikLayer *vl = vik_layers_panel_get_selected ( vlp );
577   if ( vl && vl->type == layer_type )
578   {
579     tool_func ( vl, event, vvp );
580     return TRUE;
581   }
582   else if ( VIK_LAYER(vlp->toplayer)->visible &&
583       vik_aggregate_layer_tool ( vlp->toplayer, layer_type, tool_func, event, vvp ) != 1 ) /* either accepted or rejected, but a layer was found */
584     return TRUE;
585   return FALSE;
586 }
587 #endif
588
589 VikLayer *vik_layers_panel_get_layer_of_type ( VikLayersPanel *vlp, gint type )
590 {
591   VikLayer *rv = vik_layers_panel_get_selected ( vlp );
592   if ( rv == NULL || rv->type != type )
593     if ( VIK_LAYER(vlp->toplayer)->visible )
594       return vik_aggregate_layer_get_top_visible_layer_of_type ( vlp->toplayer, type );
595     else
596       return NULL;
597   else
598     return rv;
599 }
600
601 GList *vik_layers_panel_get_all_layers_of_type(VikLayersPanel *vlp, gint type)
602 {
603   GList *layers = NULL;
604
605   return (vik_aggregate_layer_get_all_layers_of_type ( vlp->toplayer, layers, type ));
606 }
607
608 VikAggregateLayer *vik_layers_panel_get_top_layer ( VikLayersPanel *vlp )
609 {
610   return vlp->toplayer;
611 }
612
613 void vik_layers_panel_clear ( VikLayersPanel *vlp )
614 {
615   if ( (! vik_aggregate_layer_is_empty(vlp->toplayer)) && a_dialog_overwrite ( VIK_GTK_WINDOW_FROM_WIDGET(vlp), "Are you sure you wish to delete all layers?", NULL ) )
616     vik_aggregate_layer_clear ( vlp->toplayer ); /* simply deletes all layers */
617 }
618
619 void vik_layers_panel_change_coord_mode ( VikLayersPanel *vlp, VikCoordMode mode )
620 {
621   vik_layer_change_coord_mode ( VIK_LAYER(vlp->toplayer), mode );
622 }
623
624 static void layers_panel_finalize ( GObject *gob )
625 {
626   VikLayersPanel *vlp = VIK_LAYERS_PANEL ( gob );
627   g_object_unref ( VIK_LAYER(vlp->toplayer) );
628   g_object_unref ( G_OBJECT(vlp->popup_factory) );
629   G_OBJECT_CLASS(parent_class)->finalize(gob);
630 }
631