#include <string.h>
#include <glib/gi18n.h>
+#include <gdk/gdkkeysyms.h>
enum {
VLP_UPDATE_SIGNAL,
VLP_LAST_SIGNAL
};
-static void layers_panel_class_init ( VikLayersPanelClass *klass );
-static void layers_panel_init ( VikLayersPanel *vlp );
-
static guint layers_panel_signals[VLP_LAST_SIGNAL] = { 0 };
static GObjectClass *parent_class;
static void layers_popup_cb (VikLayersPanel *vlp);
static void layers_popup ( VikLayersPanel *vlp, GtkTreeIter *iter, gint mouse_button );
static gboolean layers_button_press_cb (VikLayersPanel *vlp, GdkEventButton *event);
+static gboolean layers_key_press_cb (VikLayersPanel *vlp, GdkEventKey *event);
static void layers_move_item ( VikLayersPanel *vlp, gboolean up );
static void layers_move_item_up ( VikLayersPanel *vlp );
static void layers_move_item_down ( VikLayersPanel *vlp );
static void layers_panel_finalize ( GObject *gob );
-GType vik_layers_panel_get_type()
-{
- static GType vlp_type = 0;
-
- if (!vlp_type)
- {
- static const GTypeInfo vlp_info =
- {
- sizeof (VikLayersPanelClass),
- NULL, /* base_init */
- NULL, /* base_finalize */
- (GClassInitFunc) layers_panel_class_init,
- NULL, /* class_finalize */
- NULL, /* class_data */
- sizeof (VikLayersPanel),
- 0,
- (GInstanceInitFunc) layers_panel_init,
- };
- vlp_type = g_type_register_static ( GTK_TYPE_VBOX, "VikLayersPanel", &vlp_info, 0 );
- }
-
- return vlp_type;
-}
+G_DEFINE_TYPE (VikLayersPanel, vik_layers_panel, GTK_TYPE_VBOX)
-static void layers_panel_class_init ( VikLayersPanelClass *klass )
+static void vik_layers_panel_class_init ( VikLayersPanelClass *klass )
{
GObjectClass *object_class;
return vlp->vvp;
}
-static void layers_panel_init ( VikLayersPanel *vlp )
+static void vik_layers_panel_init ( VikLayersPanel *vlp )
{
GtkWidget *hbox;
GtkWidget *addbutton, *addimage;
g_signal_connect_swapped ( vlp->vt, "button_press_event", G_CALLBACK(layers_button_press_cb), vlp);
g_signal_connect_swapped ( vlp->vt, "item_toggled", G_CALLBACK(layers_item_toggled), vlp);
g_signal_connect_swapped ( vlp->vt, "item_edited", G_CALLBACK(layers_item_edited), vlp);
+ g_signal_connect_swapped ( vlp->vt, "key_press_event", G_CALLBACK(layers_key_press_cb), vlp);
/* Add button */
addimage = gtk_image_new_from_stock ( GTK_STOCK_ADD, GTK_ICON_SIZE_SMALL_TOOLBAR );
gtk_box_pack_start ( GTK_BOX(vlp), hbox, FALSE, FALSE, 0 );
vlp->popup_factory = gtk_item_factory_new ( GTK_TYPE_MENU, "<main>", NULL );
+ gtk_item_factory_set_translate_func (vlp->popup_factory,
+ (GtkTranslateFunc) gettext, NULL, NULL);
gtk_item_factory_create_items ( vlp->popup_factory, NUM_BASE_ENTRIES, base_entries, vlp );
for ( i = 0; i < VIK_LAYER_NUM_TYPES; i++ )
{
/* TODO: FIXME: if name has a '/' in it it will get all messed up. why not have an itemfactory field with
name, icon, shortcut, etc.? */
- entry.path = g_strdup_printf("%s/New %s Layer", base_entries[NUM_BASE_ENTRIES-1].path, vik_layer_get_interface(i)->name );
+ gchar *label = g_strdup_printf(_("New _%s Layer"), vik_layer_get_interface(i)->name );
+ entry.path = g_strdup_printf("%s/%s", base_entries[NUM_BASE_ENTRIES-1].path, label );
+ g_free ( label );
entry.accelerator = NULL;
entry.callback = (GtkItemFactoryCallback) vik_layers_panel_new_layer;
entry.callback_action = i;
}
}
-void vik_layers_panel_emit_update ( VikLayersPanel *vlp )
+/**
+ * Invoke the actual drawing via signal method
+ */
+static gboolean idle_draw_panel ( VikLayersPanel *vlp )
{
g_signal_emit ( G_OBJECT(vlp), layers_panel_signals[VLP_UPDATE_SIGNAL], 0 );
+ return FALSE; // Nothing else to do
+}
+
+void vik_layers_panel_emit_update ( VikLayersPanel *vlp )
+{
+ // Only ever draw when there is time to do so
+ if ( g_thread_self() != vik_window_get_thread (VIK_WINDOW(VIK_GTK_WINDOW_FROM_WIDGET(vlp))) )
+ // Drawing requested from another (background) thread, so handle via the gdk thread method
+ gdk_threads_add_idle ( (GSourceFunc) idle_draw_panel, vlp );
+ else
+ g_idle_add ( (GSourceFunc) idle_draw_panel, vlp );
}
static void layers_item_toggled (VikLayersPanel *vlp, GtkTreeIter *iter)
{
if (event->button == 3)
{
- GtkTreeIter iter;
+ static GtkTreeIter iter;
if ( vik_treeview_get_iter_at_pos ( vlp->vt, &iter, event->x, event->y ) )
{
layers_popup ( vlp, &iter, 3 );
return FALSE;
}
+static gboolean layers_key_press_cb ( VikLayersPanel *vlp, GdkEventKey *event )
+{
+ // Accept all forms of delete keys
+ if (event->keyval == GDK_Delete || event->keyval == GDK_KP_Delete || event->keyval == GDK_BackSpace) {
+ vik_layers_panel_delete_selected (vlp);
+ return TRUE;
+ }
+ return FALSE;
+}
+
static void layers_popup ( VikLayersPanel *vlp, GtkTreeIter *iter, gint mouse_button )
{
GtkMenu *menu = NULL;
else
{
menu = GTK_MENU ( gtk_menu_new () );
- 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 ) )
+ 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 ) )
{
gtk_widget_destroy ( GTK_WIDGET(menu) );
return;
layers_popup ( vlp, NULL, 0 );
}
+/**
+ * vik_layers_panel_new_layer:
+ * @type: type of the new layer
+ *
+ * Create a new layer and add to panel.
+ */
gboolean vik_layers_panel_new_layer ( VikLayersPanel *vlp, gint type )
{
VikLayer *l;
if ( l )
{
vik_layers_panel_add_layer ( vlp, l );
- vik_layers_panel_emit_update ( vlp );
return TRUE;
}
return FALSE;
}
+/**
+ * vik_layers_panel_add_layer:
+ * @l: existing layer
+ *
+ * Add an existing layer to panel.
+ */
void vik_layers_panel_add_layer ( VikLayersPanel *vlp, VikLayer *l )
{
GtkTreeIter iter;
else
vik_aggregate_layer_add_layer ( addtoagg, l );
}
+
+ vik_layers_panel_emit_update ( vlp );
}
static void layers_move_item ( VikLayersPanel *vlp, gboolean up )
if ( ! vik_treeview_get_selected_iter ( vlp->vt, &iter ) )
return;
- vik_treeview_select_iter ( vlp->vt, &iter ); /* cancel any layer-name editing going on... */
+ vik_treeview_select_iter ( vlp->vt, &iter, FALSE ); /* cancel any layer-name editing going on... */
if ( vik_treeview_item_get_type ( vlp->vt, &iter ) == VIK_TREEVIEW_TYPE_LAYER )
{
vik_aggregate_layer_draw ( vlp->toplayer, vlp->vvp );
}
-void vik_layers_panel_draw_all_using_viewport ( VikLayersPanel *vlp, VikViewport *vvp )
-{
- if ( vlp->vvp && VIK_LAYER(vlp->toplayer)->visible )
- vik_aggregate_layer_draw ( vlp->toplayer, vvp );
-}
-
void vik_layers_panel_cut_selected ( VikLayersPanel *vlp )
{
gint type;
GtkTreeIter iter;
- g_debug(__FUNCTION__);
if ( ! vik_treeview_get_selected_iter ( vlp->vt, &iter ) )
/* Nothing to do */
return;
else
a_dialog_info_msg ( VIK_GTK_WINDOW_FROM_WIDGET(vlp), _("You cannot cut the Top Layer.") );
}
+ else if (type == VIK_TREEVIEW_TYPE_SUBLAYER) {
+ VikLayer *sel = vik_layers_panel_get_selected ( vlp );
+ if ( vik_layer_get_interface(sel->type)->cut_item ) {
+ gint subtype = vik_treeview_item_get_data( vlp->vt, &iter);
+ vik_layer_get_interface(sel->type)->cut_item ( sel, subtype, vik_treeview_item_get_pointer(sel->vt, &iter) );
+ }
+ }
}
void vik_layers_panel_copy_selected ( VikLayersPanel *vlp )
{
- gint type;
GtkTreeIter iter;
-
if ( ! vik_treeview_get_selected_iter ( vlp->vt, &iter ) )
/* Nothing to do */
return;
-
- type = vik_treeview_item_get_type ( vlp->vt, &iter );
-
- if ( type == VIK_TREEVIEW_TYPE_LAYER ) {
- a_clipboard_copy_selected ( vlp );
- }
+ // NB clipboard contains layer vs sublayer logic, so don't need to do it here
+ a_clipboard_copy_selected ( vlp );
}
void vik_layers_panel_paste_selected ( VikLayersPanel *vlp )
if ( type == VIK_TREEVIEW_TYPE_LAYER )
{
+ // Get confirmation from the user
+ if ( ! a_dialog_yes_or_no ( VIK_GTK_WINDOW_FROM_WIDGET(vlp),
+ _("Are you sure you want to delete %s?"),
+ vik_layer_get_name ( VIK_LAYER(vik_treeview_item_get_pointer ( vlp->vt, &iter )) ) ) )
+ return;
+
VikAggregateLayer *parent = vik_treeview_item_get_parent ( vlp->vt, &iter );
if ( parent )
{
return rv;
}
-GList *vik_layers_panel_get_all_layers_of_type(VikLayersPanel *vlp, gint type)
+GList *vik_layers_panel_get_all_layers_of_type(VikLayersPanel *vlp, gint type, gboolean include_invisible)
{
GList *layers = NULL;
- return (vik_aggregate_layer_get_all_layers_of_type ( vlp->toplayer, layers, type ));
+ return (vik_aggregate_layer_get_all_layers_of_type ( vlp->toplayer, layers, type, include_invisible));
}
VikAggregateLayer *vik_layers_panel_get_top_layer ( VikLayersPanel *vlp )
void vik_layers_panel_clear ( VikLayersPanel *vlp )
{
- 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 ) )
+ 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 ) )
vik_aggregate_layer_clear ( vlp->toplayer ); /* simply deletes all layers */
}
G_OBJECT_CLASS(parent_class)->finalize(gob);
}
+VikTreeview *vik_layers_panel_get_treeview ( VikLayersPanel *vlp )
+{
+ return vlp->vt;
+}