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