mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager.git
synced 2025-12-28 20:20:09 +01:00
* gnome/applet/applet-dbus-vpn.c gnome/applet/applet.c gnome/applet/applet.h - Fix up active VPN handling so that we reliably know when a VPN connection has been deactivated * src/vpn-manager/nm-vpn-manager.c - Remove duplicate VPNConnectionChange signal git-svn-id: http://svn-archive.gnome.org/svn/NetworkManager/trunk@593 4912f4e0-d625-0410-9fb7-b9a5a253dbdc
2009 lines
56 KiB
C
2009 lines
56 KiB
C
/* -*- Mode: C; tab-width: 5; indent-tabs-mode: t; c-basic-offset: 5 -*- */
|
|
/* NetworkManager Wireless Applet -- Display wireless access points and allow user control
|
|
*
|
|
* Dan Williams <dcbw@redhat.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*
|
|
* This applet used the GNOME Wireless Applet as a skeleton to build from.
|
|
*
|
|
* GNOME Wireless Applet Authors:
|
|
* Eskil Heyn Olsen <eskil@eskil.dk>
|
|
* Bastien Nocera <hadess@hadess.net> (Gnome2 port)
|
|
*
|
|
* (C) Copyright 2004-2005 Red Hat, Inc.
|
|
* (C) Copyright 2001, 2002 Free Software Foundation
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include <ctype.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <math.h>
|
|
#include <dirent.h>
|
|
#include <time.h>
|
|
|
|
#include <glib/gi18n.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#if !GTK_CHECK_VERSION(2,6,0)
|
|
#include <gnome.h>
|
|
#endif
|
|
|
|
#include <glade/glade.h>
|
|
#include <gconf/gconf-client.h>
|
|
|
|
#include "applet.h"
|
|
#include "applet-dbus.h"
|
|
#include "applet-dbus-devices.h"
|
|
#include "applet-dbus-vpn.h"
|
|
#include "other-network-dialog.h"
|
|
#include "passphrase-dialog.h"
|
|
#include "menu-items.h"
|
|
#include "vpn-password-dialog.h"
|
|
#include "vpn-connection.h"
|
|
|
|
/* Compat for GTK 2.4 and lower... */
|
|
#if (GTK_MAJOR_VERSION <= 2 && GTK_MINOR_VERSION < 6)
|
|
#define GTK_STOCK_MEDIA_PAUSE GTK_STOCK_STOP
|
|
#define GTK_STOCK_MEDIA_PLAY GTK_STOCK_REFRESH
|
|
#define GTK_STOCK_ABOUT GTK_STOCK_DIALOG_INFO
|
|
#endif
|
|
|
|
static GObject * nmwa_constructor (GType type, guint n_props, GObjectConstructParam *construct_props);
|
|
static void setup_stock (void);
|
|
static void nmwa_icons_init (NMWirelessApplet *applet);
|
|
static void nmwa_icons_free (NMWirelessApplet *applet);
|
|
static void nmwa_about_cb (NMWirelessApplet *applet);
|
|
static void nmwa_context_menu_update (NMWirelessApplet *applet);
|
|
static GtkWidget * nmwa_get_instance (NMWirelessApplet *applet);
|
|
|
|
G_DEFINE_TYPE(NMWirelessApplet, nmwa, EGG_TYPE_TRAY_ICON)
|
|
|
|
/*
|
|
* nm_null_safe_strcmp
|
|
*
|
|
* Doesn't freaking segfault if s1/s2 are NULL
|
|
*
|
|
*/
|
|
int nm_null_safe_strcmp (const char *s1, const char *s2)
|
|
{
|
|
if (!s1 && !s2)
|
|
return 0;
|
|
if (!s1 && s2)
|
|
return -1;
|
|
if (s1 && !s2)
|
|
return 1;
|
|
|
|
return (strcmp (s1, s2));
|
|
}
|
|
|
|
|
|
static void nmwa_init (NMWirelessApplet *applet)
|
|
{
|
|
applet->animation_id = 0;
|
|
applet->animation_step = 0;
|
|
glade_gnome_init ();
|
|
|
|
setup_stock ();
|
|
nmwa_icons_init (applet);
|
|
|
|
/* gtk_window_set_default_icon_from_file (ICONDIR"/NMWirelessApplet/wireless-applet.png", NULL); */
|
|
gtk_widget_show (nmwa_get_instance (applet));
|
|
}
|
|
|
|
static void nmwa_class_init (NMWirelessAppletClass *klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
|
|
gobject_class = G_OBJECT_CLASS (klass);
|
|
gobject_class->constructor = nmwa_constructor;
|
|
}
|
|
|
|
static GObject *nmwa_constructor (GType type, guint n_props, GObjectConstructParam *construct_props)
|
|
{
|
|
GObject *obj;
|
|
NMWirelessApplet *applet;
|
|
NMWirelessAppletClass *klass;
|
|
|
|
klass = NM_WIRELESS_APPLET_CLASS (g_type_class_peek (type));
|
|
obj = G_OBJECT_CLASS (nmwa_parent_class)->constructor (type, n_props, construct_props);
|
|
applet = NM_WIRELESS_APPLET (obj);
|
|
|
|
return obj;
|
|
}
|
|
|
|
|
|
void nmwa_about_cb (NMWirelessApplet *applet)
|
|
{
|
|
GdkPixbuf *pixbuf;
|
|
char *file;
|
|
GtkWidget *about_dialog;
|
|
|
|
static const gchar *authors[] =
|
|
{
|
|
"The Red Hat Desktop Team, including:\n",
|
|
"Jonathan Blandford <jrb@redhat.com>",
|
|
"John Palmieri <johnp@redhat.com>",
|
|
"Ray Strode <rstrode@redhat.com>",
|
|
"Colin Walters <walters@redhat.com>",
|
|
"Dan Williams <dcbw@redhat.com>",
|
|
"\nAnd others, including:\n",
|
|
"Bill Moss",
|
|
"Tom Parker",
|
|
"j@bootlab.org",
|
|
"Peter Jones <pjones@redhat.com>",
|
|
NULL
|
|
};
|
|
|
|
static const gchar *documenters[] =
|
|
{
|
|
NULL
|
|
};
|
|
|
|
#if !GTK_CHECK_VERSION(2,6,0)
|
|
/* GTK 2.4 and earlier, have to use libgnome for about dialog */
|
|
file = gnome_program_locate_file (NULL, GNOME_FILE_DOMAIN_PIXMAP, "gnome-networktool.png", FALSE, NULL);
|
|
pixbuf = gdk_pixbuf_new_from_file (file, NULL);
|
|
g_free (file);
|
|
|
|
about_dialog = gnome_about_new (_("NetworkManager Applet"),
|
|
VERSION,
|
|
_("Copyright (C) 2004-2005 Red Hat, Inc."),
|
|
_("Notification area applet for managing your network devices and connections."),
|
|
authors,
|
|
documenters,
|
|
NULL,
|
|
pixbuf);
|
|
g_object_unref (pixbuf);
|
|
|
|
gtk_window_set_screen (GTK_WINDOW (about_dialog), gtk_widget_get_screen (GTK_WIDGET (applet)));
|
|
g_signal_connect (about_dialog, "destroy", G_CALLBACK (gtk_widget_destroyed), &about_dialog);
|
|
gtk_widget_show (about_dialog);
|
|
|
|
#else
|
|
|
|
/* GTK 2.6 and later code */
|
|
gtk_show_about_dialog (NULL,
|
|
"name", _("NetworkManager Applet"),
|
|
"version", VERSION,
|
|
"copyright", _("Copyright (C) 2004-2005 Red Hat, Inc."),
|
|
"comments", _("Notification area applet for managing your network devices and connections."),
|
|
"authors", authors,
|
|
"documenters", documenters,
|
|
"translator-credits", NULL,
|
|
"logo-icon-name", GTK_STOCK_NETWORK,
|
|
NULL);
|
|
#endif
|
|
}
|
|
|
|
|
|
static void vpn_login_failure_dialog_close_cb (GtkWidget *dialog, gpointer user_data)
|
|
{
|
|
char *message;
|
|
|
|
if ((message = g_object_get_data (G_OBJECT (dialog), "message")))
|
|
{
|
|
g_object_set_data (G_OBJECT (dialog), "message", NULL);
|
|
g_free (message);
|
|
}
|
|
|
|
gtk_widget_destroy (dialog);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_show_vpn_login_failure_dialog
|
|
*
|
|
* Present the VPN login failure dialog.
|
|
*
|
|
*/
|
|
static gboolean nmwa_show_vpn_login_failure_dialog (char *message)
|
|
{
|
|
GtkWidget *dialog;
|
|
|
|
g_return_val_if_fail (message != NULL, FALSE);
|
|
|
|
dialog = gtk_message_dialog_new_with_markup (NULL, 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_OK, message, NULL);
|
|
g_signal_connect (dialog, "response", G_CALLBACK (vpn_login_failure_dialog_close_cb), NULL);
|
|
g_signal_connect (dialog, "close", G_CALLBACK (vpn_login_failure_dialog_close_cb), NULL);
|
|
g_object_set_data (G_OBJECT (dialog), "message", message);
|
|
gtk_widget_show_all (dialog);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_schedule_vpn_login_failure_dialog
|
|
*
|
|
* Schedule display of the VPN Login Failure dialog.
|
|
*
|
|
*/
|
|
void nmwa_schedule_vpn_login_failure_dialog (NMWirelessApplet *applet, const char *vpn_name, const char *error_msg)
|
|
{
|
|
char *msg;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (vpn_name != NULL);
|
|
g_return_if_fail (error_msg != NULL);
|
|
|
|
msg = g_strdup_printf (_("<span weight=\"bold\" size=\"larger\">VPN Login Failure</span>\n\nCould not start the "
|
|
"VPN connection '%s' due to a login failure.\n\nThe VPN service said: \"%s\""), vpn_name, error_msg);
|
|
g_idle_add ((GSourceFunc) nmwa_show_vpn_login_failure_dialog, msg);
|
|
}
|
|
|
|
|
|
static void vpn_login_banner_dialog_close_cb (GtkWidget *dialog, gpointer user_data)
|
|
{
|
|
char *message;
|
|
|
|
if ((message = g_object_get_data (G_OBJECT (dialog), "message")))
|
|
{
|
|
g_object_set_data (G_OBJECT (dialog), "message", NULL);
|
|
g_free (message);
|
|
}
|
|
|
|
gtk_widget_destroy (dialog);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_show_vpn_login_banner_dialog
|
|
*
|
|
* Present the VPN login banner dialog.
|
|
*
|
|
*/
|
|
static gboolean nmwa_show_vpn_login_banner_dialog (char *message)
|
|
{
|
|
GtkWidget *dialog;
|
|
|
|
g_return_val_if_fail (message != NULL, FALSE);
|
|
|
|
dialog = gtk_message_dialog_new_with_markup (NULL, 0, GTK_MESSAGE_INFO, GTK_BUTTONS_OK, message, NULL);
|
|
g_signal_connect (dialog, "response", G_CALLBACK (vpn_login_failure_dialog_close_cb), NULL);
|
|
g_signal_connect (dialog, "close", G_CALLBACK (vpn_login_failure_dialog_close_cb), NULL);
|
|
g_object_set_data (G_OBJECT (dialog), "message", message);
|
|
gtk_widget_show_all (dialog);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_schedule_vpn_login_banner_dialog
|
|
*
|
|
* Schedule display of the VPN Login Banner dialog.
|
|
*
|
|
*/
|
|
void nmwa_schedule_vpn_login_banner_dialog (NMWirelessApplet *applet, const char *vpn_name, const char *banner)
|
|
{
|
|
char *msg;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (vpn_name != NULL);
|
|
g_return_if_fail (banner != NULL);
|
|
|
|
msg = g_strdup_printf (_("<span weight=\"bold\" size=\"larger\">VPN Login Message</span>\n\n"
|
|
"VPN connection '%s' said:\n\n\"%s\""), vpn_name, banner);
|
|
g_idle_add ((GSourceFunc) nmwa_show_vpn_login_failure_dialog, msg);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_driver_notify_get_ignored_list
|
|
*
|
|
* Return list of devices for which we are supposed to ignore driver
|
|
* notifications for from GConf.
|
|
*
|
|
*/
|
|
GSList *nmwa_driver_notify_get_ignored_list (NMWirelessApplet *applet)
|
|
{
|
|
char *key;
|
|
GConfValue *value;
|
|
GSList *mac_list = NULL;
|
|
|
|
g_return_val_if_fail (applet != NULL, NULL);
|
|
g_return_val_if_fail (applet->gconf_client != NULL, NULL);
|
|
|
|
/* Get current list of access point MAC addresses for this AP from GConf */
|
|
key = g_strdup_printf ("%s/non_notify_cards", GCONF_PATH_PREFS);
|
|
value = gconf_client_get (applet->gconf_client, key, NULL);
|
|
|
|
if (value && (value->type == GCONF_VALUE_LIST) && (gconf_value_get_list_type (value) == GCONF_VALUE_STRING))
|
|
mac_list = gconf_client_get_list (applet->gconf_client, key, GCONF_VALUE_STRING, NULL);
|
|
|
|
if (value)
|
|
gconf_value_free (value);
|
|
g_free (key);
|
|
|
|
return (mac_list);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_driver_notify_is_device_ignored
|
|
*
|
|
* Look in GConf and determine whether or not we are supposed to
|
|
* ignore driver notifications for a particular device.
|
|
*
|
|
*/
|
|
gboolean nmwa_driver_notify_is_device_ignored (NMWirelessApplet *applet, NetworkDevice *dev)
|
|
{
|
|
gboolean found = FALSE;
|
|
GSList * mac_list = NULL;
|
|
GSList * elt;
|
|
const char * dev_addr;
|
|
|
|
g_return_val_if_fail (applet != NULL, TRUE);
|
|
g_return_val_if_fail (applet->gconf_client != NULL, TRUE);
|
|
g_return_val_if_fail (dev != NULL, TRUE);
|
|
|
|
dev_addr = network_device_get_address (dev);
|
|
g_return_val_if_fail (dev_addr != NULL, TRUE);
|
|
g_return_val_if_fail (strlen (dev_addr) > 0, TRUE);
|
|
|
|
mac_list = nmwa_driver_notify_get_ignored_list (applet);
|
|
|
|
/* Ensure that the MAC isn't already in the list */
|
|
for (elt = mac_list; elt; elt = g_slist_next (elt))
|
|
{
|
|
if (elt->data && !strcmp (dev_addr, elt->data))
|
|
{
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Free the list, since gconf_client_set_list deep-copies it */
|
|
g_slist_foreach (mac_list, (GFunc)g_free, NULL);
|
|
g_slist_free (mac_list);
|
|
|
|
return found;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_driver_notify_ignore_device
|
|
*
|
|
* Add a device's MAC address to the list of ones that we ignore
|
|
* in GConf. Stores user's pref for "Don't remind me".
|
|
*
|
|
*/
|
|
void nmwa_driver_notify_ignore_device (NMWirelessApplet *applet, NetworkDevice *dev)
|
|
{
|
|
gboolean found = FALSE;
|
|
GSList * new_mac_list = NULL;
|
|
GSList * elt;
|
|
const char * dev_addr;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (applet->gconf_client != NULL);
|
|
g_return_if_fail (dev != NULL);
|
|
|
|
dev_addr = network_device_get_address (dev);
|
|
g_return_if_fail (dev_addr != NULL);
|
|
g_return_if_fail (strlen (dev_addr) > 0);
|
|
|
|
new_mac_list = nmwa_driver_notify_get_ignored_list (applet);
|
|
|
|
/* Ensure that the MAC isn't already in the list */
|
|
for (elt = new_mac_list; elt; elt = g_slist_next (elt))
|
|
{
|
|
if (elt->data && !strcmp (dev_addr, elt->data))
|
|
{
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Add the new MAC address to the end of the list */
|
|
if (!found)
|
|
{
|
|
char *key = g_strdup_printf ("%s/non_notify_cards", GCONF_PATH_PREFS);
|
|
|
|
new_mac_list = g_slist_append (new_mac_list, g_strdup (dev_addr));
|
|
gconf_client_set_list (applet->gconf_client, key, GCONF_VALUE_STRING, new_mac_list, NULL);
|
|
g_free (key);
|
|
}
|
|
|
|
/* Free the list, since gconf_client_set_list deep-copies it */
|
|
g_slist_foreach (new_mac_list, (GFunc)g_free, NULL);
|
|
g_slist_free (new_mac_list);
|
|
}
|
|
|
|
gboolean nmwa_driver_notify_dialog_delete_cb (GtkWidget *widget, GdkEvent *event, gpointer user_data)
|
|
{
|
|
gtk_widget_destroy (widget);
|
|
return FALSE;
|
|
}
|
|
|
|
gboolean nmwa_driver_notify_dialog_destroy_cb (GtkWidget *widget, GdkEvent *event, gpointer user_data)
|
|
{
|
|
DriverNotifyCBData *cb_data = (DriverNotifyCBData *)(user_data);
|
|
NetworkDevice *dev;
|
|
|
|
g_return_val_if_fail (cb_data != NULL, FALSE);
|
|
g_return_val_if_fail (cb_data->xml != NULL, FALSE);
|
|
|
|
dev = cb_data->dev;
|
|
g_return_val_if_fail (dev != NULL, FALSE);
|
|
|
|
network_device_unref (dev);
|
|
|
|
g_object_unref (cb_data->xml);
|
|
g_free (cb_data);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
gboolean nmwa_driver_notify_ok_cb (GtkButton *button, gpointer user_data)
|
|
{
|
|
DriverNotifyCBData *cb_data = (DriverNotifyCBData *)(user_data);
|
|
NetworkDevice *dev;
|
|
NMWirelessApplet *applet;
|
|
GtkWidget *dialog;
|
|
GtkWidget *checkbox;
|
|
|
|
g_return_val_if_fail (cb_data != NULL, FALSE);
|
|
g_return_val_if_fail (cb_data->xml != NULL, FALSE);
|
|
|
|
dev = cb_data->dev;
|
|
g_return_val_if_fail (dev != NULL, FALSE);
|
|
|
|
applet = cb_data->applet;
|
|
g_return_val_if_fail (applet != NULL, FALSE);
|
|
|
|
checkbox = glade_xml_get_widget (cb_data->xml, "dont_remind_checkbox");
|
|
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (checkbox)))
|
|
nmwa_driver_notify_ignore_device (applet, dev);
|
|
|
|
dialog = glade_xml_get_widget (cb_data->xml, "driver_sucks_dialog");
|
|
gtk_widget_destroy (dialog);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_driver_notify
|
|
*
|
|
* Notify the user if there's some problem with the driver
|
|
* of a specific network device.
|
|
*
|
|
*/
|
|
gboolean nmwa_driver_notify (gpointer user_data)
|
|
{
|
|
DriverNotifyCBData * cb_data = (DriverNotifyCBData *)(user_data);
|
|
NetworkDevice * dev;
|
|
NMWirelessApplet * applet;
|
|
GtkDialog * dialog;
|
|
GtkLabel * label;
|
|
char * label_text = NULL;
|
|
char * temp = NULL;
|
|
GtkButton * button;
|
|
NMDriverSupportLevel support_level;
|
|
|
|
g_return_val_if_fail (cb_data != NULL, FALSE);
|
|
|
|
dev = cb_data->dev;
|
|
g_return_val_if_fail (dev != NULL, FALSE);
|
|
|
|
if (!(applet = cb_data->applet) || !applet->glade_file)
|
|
goto out;
|
|
|
|
/* If the user has already requested that we ignore notifications for
|
|
* this device, don't do anything.
|
|
*/
|
|
if (nmwa_driver_notify_is_device_ignored (applet, dev))
|
|
goto out;
|
|
|
|
if (!(cb_data->xml = glade_xml_new (applet->glade_file, "driver_sucks_dialog", NULL)))
|
|
{
|
|
nmwa_schedule_warning_dialog (applet, _("The NetworkManager Applet could not find some required resources (the glade file was not found)."));
|
|
goto out;
|
|
}
|
|
|
|
dialog = GTK_DIALOG (glade_xml_get_widget (cb_data->xml, "driver_sucks_dialog"));
|
|
g_signal_connect (G_OBJECT (dialog), "destroy-event", GTK_SIGNAL_FUNC (nmwa_driver_notify_dialog_destroy_cb), cb_data);
|
|
g_signal_connect (G_OBJECT (dialog), "delete-event", GTK_SIGNAL_FUNC (nmwa_driver_notify_dialog_delete_cb), cb_data);
|
|
|
|
label = GTK_LABEL (glade_xml_get_widget (cb_data->xml, "driver_sucks_label"));
|
|
|
|
switch (network_device_get_driver_support_level (dev))
|
|
{
|
|
case NM_DRIVER_NO_WIRELESS_SCAN:
|
|
temp = g_strdup_printf (_("The network device \"%s (%s)\" does not support wireless scanning."),
|
|
network_device_get_desc (dev), network_device_get_iface (dev));
|
|
label_text = g_strdup_printf (gtk_label_get_label (label), temp);
|
|
g_free (temp);
|
|
break;
|
|
|
|
case NM_DRIVER_NO_CARRIER_DETECT:
|
|
temp = g_strdup_printf (_("The network device \"%s (%s)\" does not support link detection."),
|
|
network_device_get_desc (dev), network_device_get_iface (dev));
|
|
label_text = g_strdup_printf (gtk_label_get_label (label), temp);
|
|
g_free (temp);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (label_text)
|
|
gtk_label_set_markup (label, label_text);
|
|
|
|
button = GTK_BUTTON (glade_xml_get_widget (cb_data->xml, "ok_button"));
|
|
g_signal_connect (G_OBJECT (button), "clicked", GTK_SIGNAL_FUNC (nmwa_driver_notify_ok_cb), cb_data);
|
|
|
|
gtk_widget_show_all (GTK_WIDGET (dialog));
|
|
|
|
out:
|
|
network_device_unref (cb_data->dev);
|
|
return (FALSE);
|
|
}
|
|
|
|
|
|
static void nmwa_set_icon (NMWirelessApplet *applet, GdkPixbuf *new_icon)
|
|
{
|
|
GdkPixbuf *composite;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (new_icon != NULL);
|
|
|
|
composite = gdk_pixbuf_copy (new_icon);
|
|
|
|
if (applet->gui_active_vpn)
|
|
{
|
|
int dest_x = gdk_pixbuf_get_width (new_icon) - gdk_pixbuf_get_width (applet->vpn_lock_icon);
|
|
int dest_y = gdk_pixbuf_get_height (new_icon) - gdk_pixbuf_get_height (applet->vpn_lock_icon) - 2;
|
|
|
|
gdk_pixbuf_composite (applet->vpn_lock_icon, composite, dest_x, dest_y, gdk_pixbuf_get_width (applet->vpn_lock_icon),
|
|
gdk_pixbuf_get_height (applet->vpn_lock_icon), dest_x, dest_y, 1.0, 1.0, GDK_INTERP_NEAREST, 255);
|
|
}
|
|
|
|
gtk_image_set_from_pixbuf (GTK_IMAGE (applet->pixmap), composite);
|
|
g_object_unref (composite);
|
|
}
|
|
|
|
|
|
/*
|
|
* animation_timeout
|
|
*
|
|
* Jump to the next frame of the applets icon if the icon
|
|
* is supposed to be animated.
|
|
*
|
|
*/
|
|
static gboolean animation_timeout (NMWirelessApplet *applet)
|
|
{
|
|
if (!applet->nm_running)
|
|
{
|
|
applet->animation_step = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
switch (applet->gui_nm_state)
|
|
{
|
|
case (NM_STATE_CONNECTING):
|
|
if (network_device_get_type (applet->gui_active_device) == DEVICE_TYPE_WIRELESS_ETHERNET)
|
|
{
|
|
if (applet->animation_step >= NUM_WIRELESS_CONNECTING_FRAMES)
|
|
applet->animation_step = 0;
|
|
nmwa_set_icon (applet, applet->wireless_connecting_icons[applet->animation_step]);
|
|
}
|
|
else
|
|
{
|
|
if (applet->animation_step >= NUM_WIRED_CONNECTING_FRAMES)
|
|
applet->animation_step = 0;
|
|
nmwa_set_icon (applet, applet->wired_connecting_icons[applet->animation_step]);
|
|
}
|
|
applet->animation_step ++;
|
|
break;
|
|
|
|
case (NM_STATE_SCANNING):
|
|
if (applet->animation_step >= NUM_WIRELESS_SCANNING_FRAMES)
|
|
applet->animation_step = 0;
|
|
nmwa_set_icon (applet, applet->wireless_scanning_icons[applet->animation_step]);
|
|
applet->animation_step ++;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_update_state
|
|
*
|
|
* Figure out what the currently active device is from NetworkManager, its type,
|
|
* and what our icon on the panel should look like for each type.
|
|
*
|
|
*/
|
|
static void nmwa_update_state (NMWirelessApplet *applet)
|
|
{
|
|
gboolean show_applet = TRUE;
|
|
gboolean need_animation = FALSE;
|
|
gboolean active_vpn = FALSE;
|
|
GdkPixbuf * pixbuf = NULL;
|
|
gint strength = -1;
|
|
char * tip = NULL;
|
|
WirelessNetwork * active_network = NULL;
|
|
|
|
g_mutex_lock (applet->data_mutex);
|
|
|
|
if (applet->gui_active_device && (network_device_get_type (applet->gui_active_device) == DEVICE_TYPE_WIRELESS_ETHERNET))
|
|
{
|
|
active_network = network_device_get_active_wireless_network (applet->gui_active_device);
|
|
strength = CLAMP ((int)network_device_get_strength (applet->gui_active_device), 0, 100);
|
|
}
|
|
|
|
if (!applet->nm_running)
|
|
{
|
|
show_applet = FALSE;
|
|
tip = g_strdup (_("NetworkManager is not running"));
|
|
goto done;
|
|
}
|
|
|
|
if (!applet->gui_active_device)
|
|
applet->gui_nm_state = NM_STATE_DISCONNECTED;
|
|
|
|
switch (applet->gui_nm_state)
|
|
{
|
|
case NM_STATE_DISCONNECTED:
|
|
pixbuf = applet->no_connection_icon;
|
|
tip = g_strdup (_("No network connection"));
|
|
break;
|
|
|
|
case NM_STATE_CONNECTED:
|
|
if (network_device_get_type (applet->gui_active_device) == DEVICE_TYPE_WIRED_ETHERNET)
|
|
{
|
|
pixbuf = applet->wired_icon;
|
|
tip = g_strdup (_("Wired network connection"));
|
|
}
|
|
else if (network_device_get_type (applet->gui_active_device) == DEVICE_TYPE_WIRELESS_ETHERNET)
|
|
{
|
|
if (applet->is_adhoc)
|
|
{
|
|
pixbuf = applet->adhoc_icon;
|
|
tip = g_strdup (_("Connected to an Ad-Hoc wireless network"));
|
|
}
|
|
else
|
|
{
|
|
if (strength > 75)
|
|
pixbuf = applet->wireless_100_icon;
|
|
else if (strength > 50)
|
|
pixbuf = applet->wireless_75_icon;
|
|
else if (strength > 25)
|
|
pixbuf = applet->wireless_50_icon;
|
|
else if (strength > 0)
|
|
pixbuf = applet->wireless_25_icon;
|
|
else
|
|
pixbuf = applet->wireless_00_icon;
|
|
tip = g_strdup_printf (_("Wireless network connection to '%s' (%d%%)"),
|
|
active_network ? wireless_network_get_essid (active_network) : "(unknown)", strength);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case NM_STATE_CONNECTING:
|
|
if (network_device_get_type (applet->gui_active_device) == DEVICE_TYPE_WIRED_ETHERNET)
|
|
tip = g_strdup (_("Connecting to a wired network..."));
|
|
else if (network_device_get_type (applet->gui_active_device) == DEVICE_TYPE_WIRELESS_ETHERNET)
|
|
tip = g_strdup_printf (_("Connecting to wireless network '%s'..."),
|
|
active_network ? wireless_network_get_essid (active_network) : "(unknown)");
|
|
need_animation = TRUE;
|
|
break;
|
|
|
|
case NM_STATE_SCANNING:
|
|
need_animation = TRUE;
|
|
tip = g_strdup (_("Scanning for wireless networks..."));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
done:
|
|
g_mutex_unlock (applet->data_mutex);
|
|
|
|
if (!applet->tooltips)
|
|
applet->tooltips = gtk_tooltips_new ();
|
|
gtk_tooltips_set_tip (applet->tooltips, applet->event_box, tip, NULL);
|
|
g_free (tip);
|
|
|
|
if (applet->animation_id)
|
|
g_source_remove (applet->animation_id);
|
|
if (need_animation)
|
|
applet->animation_id = g_timeout_add (100, (GSourceFunc) animation_timeout, applet);
|
|
else
|
|
{
|
|
if (pixbuf)
|
|
nmwa_set_icon (applet, pixbuf);
|
|
else
|
|
show_applet = FALSE;
|
|
}
|
|
|
|
/* determine if we should hide the notification icon */
|
|
if (show_applet)
|
|
gtk_widget_show (GTK_WIDGET (applet));
|
|
else
|
|
gtk_widget_hide (GTK_WIDGET (applet));
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_redraw_timeout
|
|
*
|
|
* Called regularly to update the applet's state and icon in the panel
|
|
*
|
|
*/
|
|
static int nmwa_redraw_timeout (NMWirelessApplet *applet)
|
|
{
|
|
nmwa_update_state (applet);
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
static void nmwa_start_redraw_timeout (NMWirelessApplet *applet)
|
|
{
|
|
applet->redraw_timeout_id = g_timeout_add (1000, (GtkFunction) nmwa_redraw_timeout, applet);
|
|
}
|
|
|
|
|
|
/*
|
|
* show_warning_dialog
|
|
*
|
|
* pop up a warning or error dialog with certain text
|
|
*
|
|
*/
|
|
gboolean show_warning_dialog (gchar *mesg)
|
|
{
|
|
GtkWidget * dialog;
|
|
|
|
dialog = gtk_message_dialog_new (NULL, 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_OK, mesg, NULL);
|
|
gtk_dialog_run (GTK_DIALOG (dialog));
|
|
gtk_widget_destroy (dialog);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_schedule_warning_dialog
|
|
*
|
|
* Run a warning dialog in the main event loop.
|
|
*
|
|
*/
|
|
void nmwa_schedule_warning_dialog (NMWirelessApplet *applet, const char *msg)
|
|
{
|
|
char *lcl_msg;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (msg != NULL);
|
|
|
|
lcl_msg = g_strdup (msg);
|
|
g_idle_add ((GSourceFunc) nmwa_show_vpn_login_failure_dialog, lcl_msg);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_update_network_timestamp
|
|
*
|
|
* Update the timestamp of a network in GConf.
|
|
*
|
|
*/
|
|
static void nmwa_update_network_timestamp (NMWirelessApplet *applet, WirelessNetwork *network)
|
|
{
|
|
char * key;
|
|
char * escaped_network;
|
|
const char * net_essid;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (network != NULL);
|
|
|
|
net_essid = wireless_network_get_essid (network);
|
|
|
|
/* Update GConf to set timestamp for this network, or add it if
|
|
* it doesn't already exist.
|
|
*/
|
|
|
|
/* Update timestamp on network */
|
|
escaped_network = gconf_escape_key (net_essid, strlen (net_essid));
|
|
key = g_strdup_printf ("%s/%s/timestamp", GCONF_PATH_WIRELESS_NETWORKS, escaped_network);
|
|
gconf_client_set_int (applet->gconf_client, key, time (NULL), NULL);
|
|
g_free (key);
|
|
|
|
/* Force-set the essid too so that we have a semi-complete network entry */
|
|
key = g_strdup_printf ("%s/%s/essid", GCONF_PATH_WIRELESS_NETWORKS, escaped_network);
|
|
gconf_client_set_string (applet->gconf_client, key, net_essid, NULL);
|
|
g_free (key);
|
|
g_free (escaped_network);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_get_device_for_nm_device
|
|
*
|
|
* Searches the device list for a device that matches the
|
|
* NetworkManager ID given.
|
|
*
|
|
*/
|
|
NetworkDevice *nmwa_get_device_for_nm_device (GSList *dev_list, const char *nm_dev)
|
|
{
|
|
NetworkDevice *found_dev = NULL;
|
|
GSList *elt;
|
|
|
|
g_return_val_if_fail (nm_dev != NULL, NULL);
|
|
g_return_val_if_fail (strlen (nm_dev), NULL);
|
|
|
|
for (elt = dev_list; elt; elt = g_slist_next (elt))
|
|
{
|
|
NetworkDevice *dev = (NetworkDevice *)(elt->data);
|
|
if (dev && (strcmp (network_device_get_nm_path (dev), nm_dev) == 0))
|
|
{
|
|
found_dev = dev;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return (found_dev);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_item_activate
|
|
*
|
|
* Signal function called when user clicks on a menu item
|
|
*
|
|
*/
|
|
static void nmwa_menu_item_activate (GtkMenuItem *item, gpointer user_data)
|
|
{
|
|
NMWirelessApplet *applet = (NMWirelessApplet *)user_data;
|
|
NetworkDevice *dev = NULL;
|
|
WirelessNetwork *net = NULL;
|
|
char *tag;
|
|
|
|
g_return_if_fail (item != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (item), "network")))
|
|
{
|
|
char *item_dev = g_object_get_data (G_OBJECT (item), "nm_device");
|
|
|
|
g_mutex_lock (applet->data_mutex);
|
|
if (item_dev && (dev = nmwa_get_device_for_nm_device (applet->gui_device_list, item_dev)))
|
|
{
|
|
network_device_ref (dev);
|
|
g_mutex_unlock (applet->data_mutex);
|
|
if ((net = network_device_get_wireless_network_by_essid (dev, tag)))
|
|
nmwa_update_network_timestamp (applet, net);
|
|
}
|
|
else
|
|
g_mutex_unlock (applet->data_mutex);
|
|
}
|
|
else if ((tag = g_object_get_data (G_OBJECT (item), "device")))
|
|
{
|
|
g_mutex_lock (applet->data_mutex);
|
|
dev = nmwa_get_device_for_nm_device (applet->gui_device_list, tag);
|
|
network_device_ref (dev);
|
|
g_mutex_unlock (applet->data_mutex);
|
|
}
|
|
|
|
if (dev)
|
|
{
|
|
nmwa_dbus_set_device (applet->connection, dev, wireless_network_get_essid (net), -1, NULL);
|
|
network_device_unref (dev);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_vpn_item_activate
|
|
*
|
|
* Signal function called when user clicks on a VPN menu item
|
|
*
|
|
*/
|
|
static void nmwa_menu_vpn_item_activate (GtkMenuItem *item, gpointer user_data)
|
|
{
|
|
NMWirelessApplet *applet = (NMWirelessApplet *)user_data;
|
|
char *tag;
|
|
|
|
g_return_if_fail (item != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (item), "vpn")))
|
|
{
|
|
VPNConnection *vpn = (VPNConnection *)tag;
|
|
const char *name = nmwa_vpn_connection_get_name (vpn);
|
|
char *password = NULL;
|
|
|
|
if (vpn != applet->gui_active_vpn)
|
|
{
|
|
if ((password = nmwa_vpn_request_password (applet, name, nmwa_vpn_connection_get_user_name (vpn), FALSE)))
|
|
{
|
|
nmwa_dbus_vpn_activate_connection (applet->connection, name, password);
|
|
g_free (password);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_disconnect_vpn_item_activate
|
|
*
|
|
* Signal function called when user clicks on a VPN menu item
|
|
*
|
|
*/
|
|
static void nmwa_menu_disconnect_vpn_item_activate (GtkMenuItem *item, gpointer user_data)
|
|
{
|
|
NMWirelessApplet *applet = (NMWirelessApplet *)user_data;
|
|
|
|
g_return_if_fail (item != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
nmwa_dbus_vpn_deactivate_connection (applet->connection);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_add_separator_item
|
|
*
|
|
*/
|
|
static void nmwa_menu_add_separator_item (GtkWidget *menu)
|
|
{
|
|
GtkWidget *menu_item;
|
|
menu_item = gtk_separator_menu_item_new ();
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
gtk_widget_show (menu_item);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_add_text_item
|
|
*
|
|
* Add a non-clickable text item to a menu
|
|
*
|
|
*/
|
|
static void nmwa_menu_add_text_item (GtkWidget *menu, char *text)
|
|
{
|
|
GtkWidget *menu_item;
|
|
|
|
g_return_if_fail (text != NULL);
|
|
g_return_if_fail (menu != NULL);
|
|
|
|
menu_item = gtk_menu_item_new_with_label (text);
|
|
gtk_widget_set_sensitive (menu_item, FALSE);
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
gtk_widget_show (menu_item);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_add_device_item
|
|
*
|
|
* Add a network device to the menu
|
|
*
|
|
*/
|
|
static void nmwa_menu_add_device_item (GtkWidget *menu, NetworkDevice *device, gboolean current, gint n_devices, NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget * menu_item;
|
|
|
|
g_return_if_fail (menu != NULL);
|
|
g_return_if_fail (device != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
switch (network_device_get_type (device))
|
|
{
|
|
case DEVICE_TYPE_WIRED_ETHERNET:
|
|
{
|
|
NMWiredMenuItem *item = wired_menu_item_new ();
|
|
GtkCheckMenuItem *gtk_item = wired_menu_item_get_check_item (item);
|
|
wired_menu_item_update (item, device, n_devices);
|
|
if (applet->gui_active_device == device)
|
|
gtk_check_menu_item_set_active (gtk_item, TRUE);
|
|
|
|
g_object_set_data (G_OBJECT (gtk_item), "device", g_strdup (network_device_get_iface (device)));
|
|
g_object_set_data (G_OBJECT (gtk_item), "nm-item-data", item);
|
|
g_signal_connect(G_OBJECT (gtk_item), "activate", G_CALLBACK (nmwa_menu_item_activate), applet);
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), GTK_WIDGET (gtk_item));
|
|
gtk_widget_show (GTK_WIDGET (gtk_item));
|
|
break;
|
|
}
|
|
|
|
case DEVICE_TYPE_WIRELESS_ETHERNET:
|
|
{
|
|
NMWirelessMenuItem *item = wireless_menu_item_new ();
|
|
GtkMenuItem *gtk_item = wireless_menu_item_get_item (item);
|
|
wireless_menu_item_update (item, device, n_devices);
|
|
|
|
g_object_set_data (G_OBJECT (gtk_item), "device", g_strdup (network_device_get_iface (device)));
|
|
g_object_set_data (G_OBJECT (gtk_item), "nm-item-data", item);
|
|
g_signal_connect(G_OBJECT (gtk_item), "activate", G_CALLBACK (nmwa_menu_item_activate), applet);
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), GTK_WIDGET (gtk_item));
|
|
gtk_widget_show (GTK_WIDGET (gtk_item));
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void custom_essid_item_selected (GtkWidget *menu_item, NMWirelessApplet *applet)
|
|
{
|
|
nmwa_other_network_dialog_run (applet, FALSE);
|
|
}
|
|
|
|
|
|
static void nmwa_menu_add_custom_essid_item (GtkWidget *menu, NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget *menu_item;
|
|
GtkWidget *label;
|
|
|
|
menu_item = gtk_menu_item_new ();
|
|
label = gtk_label_new (_("Other Wireless Networks..."));
|
|
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
|
|
gtk_container_add (GTK_CONTAINER (menu_item), label);
|
|
gtk_widget_show_all (menu_item);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
g_signal_connect (menu_item, "activate", G_CALLBACK (custom_essid_item_selected), applet);
|
|
}
|
|
|
|
|
|
static void new_network_item_selected (GtkWidget *menu_item, NMWirelessApplet *applet)
|
|
{
|
|
nmwa_other_network_dialog_run (applet, TRUE);
|
|
}
|
|
|
|
|
|
static void nmwa_menu_add_create_network_item (GtkWidget *menu, NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget *menu_item;
|
|
GtkWidget *label;
|
|
|
|
menu_item = gtk_menu_item_new ();
|
|
label = gtk_label_new (_("Create new Wireless Network..."));
|
|
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
|
|
gtk_container_add (GTK_CONTAINER (menu_item), label);
|
|
gtk_widget_show_all (menu_item);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
g_signal_connect (menu_item, "activate", G_CALLBACK (new_network_item_selected), applet);
|
|
}
|
|
|
|
|
|
typedef struct AddNetworksCB
|
|
{
|
|
NMWirelessApplet * applet;
|
|
gboolean has_encrypted;
|
|
GtkWidget * menu;
|
|
} AddNetworksCB;
|
|
|
|
|
|
/*
|
|
* nmwa_add_networks_helper
|
|
*
|
|
*/
|
|
void nmwa_add_networks_helper (NetworkDevice *dev, WirelessNetwork *net, gpointer user_data)
|
|
{
|
|
AddNetworksCB * cb_data = (AddNetworksCB *)user_data;
|
|
NMNetworkMenuItem * item;
|
|
GtkCheckMenuItem * gtk_item;
|
|
|
|
g_return_if_fail (dev != NULL);
|
|
g_return_if_fail (net != NULL);
|
|
g_return_if_fail (cb_data != NULL);
|
|
g_return_if_fail (cb_data->menu != NULL);
|
|
g_return_if_fail (cb_data->applet != NULL);
|
|
|
|
item = network_menu_item_new (cb_data->applet->encryption_size_group);
|
|
gtk_item = network_menu_item_get_check_item (item);
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (cb_data->menu), GTK_WIDGET (gtk_item));
|
|
if (cb_data->applet->gui_active_device == dev && wireless_network_get_active (net))
|
|
gtk_check_menu_item_set_active (gtk_item, TRUE);
|
|
network_menu_item_update (item, net, cb_data->has_encrypted);
|
|
|
|
g_object_set_data (G_OBJECT (gtk_item), "network", g_strdup (wireless_network_get_essid (net)));
|
|
g_object_set_data (G_OBJECT (gtk_item), "nm_device", g_strdup (network_device_get_nm_path (dev)));
|
|
g_object_set_data (G_OBJECT (gtk_item), "nm-item-data", item);
|
|
g_signal_connect (G_OBJECT (gtk_item), "activate", G_CALLBACK (nmwa_menu_item_activate), cb_data->applet);
|
|
|
|
gtk_widget_show (GTK_WIDGET (gtk_item));
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_has_encrypted_networks_helper
|
|
*
|
|
*/
|
|
void nmwa_has_encrypted_networks_helper (NetworkDevice *dev, WirelessNetwork *net, gpointer user_data)
|
|
{
|
|
gboolean * has_encrypted = user_data;
|
|
|
|
g_return_if_fail (dev != NULL);
|
|
g_return_if_fail (net != NULL);
|
|
g_return_if_fail (has_encrypted != NULL);
|
|
|
|
if (wireless_network_get_encrypted (net))
|
|
*has_encrypted = TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_device_add_networks
|
|
*
|
|
*/
|
|
static void nmwa_menu_device_add_networks (GtkWidget *menu, NetworkDevice *dev, NMWirelessApplet *applet)
|
|
{
|
|
GSList * list;
|
|
gboolean has_encrypted = FALSE;
|
|
AddNetworksCB * add_networks_cb = NULL;
|
|
|
|
g_return_if_fail (menu != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (dev != NULL);
|
|
|
|
if (network_device_get_type (dev) != DEVICE_TYPE_WIRELESS_ETHERNET)
|
|
return;
|
|
|
|
/* Check for any security */
|
|
network_device_foreach_wireless_network (dev, nmwa_has_encrypted_networks_helper, &has_encrypted);
|
|
|
|
add_networks_cb = g_malloc0 (sizeof (AddNetworksCB));
|
|
add_networks_cb->applet = applet;
|
|
add_networks_cb->has_encrypted = has_encrypted;
|
|
add_networks_cb->menu = menu;
|
|
|
|
/* Add all networks in our network list to the menu */
|
|
network_device_foreach_wireless_network (dev, nmwa_add_networks_helper, add_networks_cb);
|
|
|
|
g_free (add_networks_cb);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_add_devices
|
|
*
|
|
*/
|
|
static void nmwa_menu_add_vpn_menu (GtkWidget *menu, NMWirelessApplet *applet)
|
|
{
|
|
GtkMenuItem *item;
|
|
GtkMenu *vpn_menu;
|
|
GtkMenuItem *other_item;
|
|
GSList *elt;
|
|
|
|
g_return_if_fail (menu != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
item = GTK_MENU_ITEM (gtk_menu_item_new_with_label (_("VPN Connections")));
|
|
|
|
vpn_menu = GTK_MENU (gtk_menu_new ());
|
|
for (elt = applet->gui_vpn_connections; elt; elt = g_slist_next (elt))
|
|
{
|
|
GtkCheckMenuItem *vpn_item;
|
|
VPNConnection *vpn = elt->data;
|
|
const char *vpn_name = nmwa_vpn_connection_get_name (vpn);
|
|
|
|
vpn_item = GTK_CHECK_MENU_ITEM (gtk_check_menu_item_new_with_label (vpn_name));
|
|
nmwa_vpn_connection_ref (vpn);
|
|
g_object_set_data (G_OBJECT (vpn_item), "vpn", vpn);
|
|
|
|
if (applet->gui_active_vpn && (strcmp (vpn_name, nmwa_vpn_connection_get_name (applet->gui_active_vpn)) == 0))
|
|
gtk_check_menu_item_set_active (vpn_item, TRUE);
|
|
|
|
g_signal_connect (G_OBJECT (vpn_item), "activate", G_CALLBACK (nmwa_menu_vpn_item_activate), applet);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (vpn_menu), GTK_WIDGET (vpn_item));
|
|
}
|
|
other_item = GTK_MENU_ITEM (gtk_separator_menu_item_new ());
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (vpn_menu), GTK_WIDGET (other_item));
|
|
|
|
other_item = GTK_MENU_ITEM (gtk_menu_item_new_with_label (_("Disconnect VPN...")));
|
|
g_signal_connect (G_OBJECT (other_item), "activate", G_CALLBACK (nmwa_menu_disconnect_vpn_item_activate), applet);
|
|
if (!applet->gui_active_vpn)
|
|
gtk_widget_set_sensitive (GTK_WIDGET (other_item), FALSE);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (vpn_menu), GTK_WIDGET (other_item));
|
|
|
|
gtk_menu_item_set_submenu (item, GTK_WIDGET (vpn_menu));
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), GTK_WIDGET (item));
|
|
gtk_widget_show_all (GTK_WIDGET (item));
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_add_devices
|
|
*
|
|
*/
|
|
static void nmwa_menu_add_devices (GtkWidget *menu, NMWirelessApplet *applet)
|
|
{
|
|
GSList *element;
|
|
gint n_wireless_interfaces = 0;
|
|
gint n_wired_interfaces = 0;
|
|
|
|
g_return_if_fail (menu != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
g_mutex_lock (applet->data_mutex);
|
|
if (! applet->gui_device_list)
|
|
{
|
|
nmwa_menu_add_text_item (menu, _("No network devices have been found"));
|
|
g_mutex_unlock (applet->data_mutex);
|
|
return;
|
|
}
|
|
|
|
for (element = applet->gui_device_list; element; element = element->next)
|
|
{
|
|
NetworkDevice *dev = (NetworkDevice *)(element->data);
|
|
|
|
g_assert (dev);
|
|
|
|
switch (network_device_get_type (dev))
|
|
{
|
|
case DEVICE_TYPE_WIRELESS_ETHERNET:
|
|
n_wireless_interfaces++;
|
|
break;
|
|
case DEVICE_TYPE_WIRED_ETHERNET:
|
|
n_wired_interfaces++;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Add all devices in our device list to the menu */
|
|
for (element = applet->gui_device_list; element; element = element->next)
|
|
{
|
|
NetworkDevice *dev = (NetworkDevice *)(element->data);
|
|
|
|
if (dev)
|
|
{
|
|
gboolean current = (dev == applet->gui_active_device);
|
|
gint n_devices = 0;
|
|
|
|
switch (network_device_get_type (dev))
|
|
{
|
|
case DEVICE_TYPE_WIRED_ETHERNET:
|
|
n_devices = n_wired_interfaces;
|
|
break;
|
|
|
|
case DEVICE_TYPE_WIRELESS_ETHERNET:
|
|
n_devices = n_wireless_interfaces;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (n_devices >= 0)
|
|
{
|
|
nmwa_menu_add_device_item (menu, dev, current, n_devices, applet);
|
|
nmwa_menu_device_add_networks (menu, dev, applet);
|
|
}
|
|
}
|
|
}
|
|
|
|
nmwa_menu_add_separator_item (menu);
|
|
nmwa_menu_add_vpn_menu (menu, applet);
|
|
|
|
if (n_wireless_interfaces > 0)
|
|
{
|
|
/* Add the "Other wireless network..." entry */
|
|
nmwa_menu_add_separator_item (menu);
|
|
nmwa_menu_add_custom_essid_item (menu, applet);
|
|
nmwa_menu_add_create_network_item (menu, applet);
|
|
}
|
|
|
|
g_mutex_unlock (applet->data_mutex);
|
|
}
|
|
|
|
|
|
static void nmwa_set_scanning_enabled_cb (GtkWidget *widget, NMWirelessApplet *applet)
|
|
{
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
nmwa_dbus_enable_scanning (applet, !applet->scanning_enabled);
|
|
}
|
|
|
|
static void nmwa_set_wireless_enabled_cb (GtkWidget *widget, NMWirelessApplet *applet)
|
|
{
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
nmwa_dbus_enable_wireless (applet, !applet->wireless_enabled);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_menu_item_data_free
|
|
*
|
|
* Frees the "network" data tag on a menu item we've created
|
|
*
|
|
*/
|
|
static void nmwa_menu_item_data_free (GtkWidget *menu_item, gpointer data)
|
|
{
|
|
char *tag;
|
|
GtkMenu *menu;
|
|
|
|
g_return_if_fail (menu_item != NULL);
|
|
g_return_if_fail (data != NULL);
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (menu_item), "network")))
|
|
{
|
|
g_object_set_data (G_OBJECT (menu_item), "network", NULL);
|
|
g_free (tag);
|
|
}
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (menu_item), "nm_device")))
|
|
{
|
|
g_object_set_data (G_OBJECT (menu_item), "nm_device", NULL);
|
|
g_free (tag);
|
|
}
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (menu_item), "nm-item-data")))
|
|
{
|
|
g_object_set_data (G_OBJECT (menu_item), "nm-item-data", NULL);
|
|
g_free (tag);
|
|
}
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (menu_item), "device")))
|
|
{
|
|
g_object_set_data (G_OBJECT (menu_item), "device", NULL);
|
|
g_free (tag);
|
|
}
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (menu_item), "vpn")))
|
|
{
|
|
g_object_set_data (G_OBJECT (menu_item), "vpn", NULL);
|
|
nmwa_vpn_connection_unref ((VPNConnection *)tag);
|
|
}
|
|
|
|
if ((tag = g_object_get_data (G_OBJECT (menu_item), "disconnect")))
|
|
{
|
|
g_object_set_data (G_OBJECT (menu_item), "disconnect", NULL);
|
|
g_free (tag);
|
|
}
|
|
|
|
if ((menu = GTK_MENU (gtk_menu_item_get_submenu (GTK_MENU_ITEM (menu_item)))))
|
|
gtk_container_foreach (GTK_CONTAINER (menu), nmwa_menu_item_data_free, menu);
|
|
|
|
gtk_widget_destroy (menu_item);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_dispose_menu_items
|
|
*
|
|
* Destroy the menu and each of its items data tags
|
|
*
|
|
*/
|
|
static void nmwa_dropdown_menu_clear (GtkWidget *menu)
|
|
{
|
|
g_return_if_fail (menu != NULL);
|
|
|
|
/* Free the "network" data on each menu item, and destroy the item */
|
|
gtk_container_foreach (GTK_CONTAINER (menu), nmwa_menu_item_data_free, menu);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_dropdown_menu_populate
|
|
*
|
|
* Set up our networks menu from scratch
|
|
*
|
|
*/
|
|
static void nmwa_dropdown_menu_populate (GtkWidget *menu, NMWirelessApplet *applet)
|
|
{
|
|
g_return_if_fail (menu != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if (!applet->nm_running)
|
|
nmwa_menu_add_text_item (menu, _("NetworkManager is not running..."));
|
|
else
|
|
nmwa_menu_add_devices (menu, applet);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_dropdown_menu_show_cb
|
|
*
|
|
* Pop up the wireless networks menu
|
|
*
|
|
*/
|
|
static void nmwa_dropdown_menu_show_cb (GtkWidget *menu, NMWirelessApplet *applet)
|
|
{
|
|
g_return_if_fail (menu != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if (!applet->tooltips)
|
|
applet->tooltips = gtk_tooltips_new ();
|
|
gtk_tooltips_set_tip (applet->tooltips, applet->event_box, NULL, NULL);
|
|
|
|
if (applet->dropdown_menu && (menu == applet->dropdown_menu))
|
|
{
|
|
nmwa_dropdown_menu_clear (applet->dropdown_menu);
|
|
nmwa_dropdown_menu_populate (applet->dropdown_menu, applet);
|
|
gtk_widget_show_all (applet->dropdown_menu);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* nmwa_dropdown_menu_create
|
|
*
|
|
* Create the applet's dropdown menu
|
|
*
|
|
*/
|
|
static GtkWidget *nmwa_dropdown_menu_create (GtkMenuItem *parent, NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget *menu;
|
|
|
|
g_return_val_if_fail (parent != NULL, NULL);
|
|
g_return_val_if_fail (applet != NULL, NULL);
|
|
|
|
menu = gtk_menu_new ();
|
|
gtk_container_set_border_width (GTK_CONTAINER (menu), 0);
|
|
gtk_menu_item_set_submenu (GTK_MENU_ITEM (parent), menu);
|
|
g_signal_connect (menu, "show", G_CALLBACK (nmwa_dropdown_menu_show_cb), applet);
|
|
|
|
return menu;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_context_menu_update
|
|
*
|
|
*/
|
|
static void nmwa_context_menu_update (NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget *image;
|
|
|
|
g_return_if_fail (applet != NULL);
|
|
g_return_if_fail (applet->pause_scanning_item != NULL);
|
|
g_return_if_fail (applet->stop_wireless_item != NULL);
|
|
|
|
g_mutex_lock (applet->data_mutex);
|
|
|
|
gtk_widget_destroy (applet->pause_scanning_item);
|
|
gtk_widget_destroy (applet->stop_wireless_item);
|
|
|
|
if (applet->scanning_enabled)
|
|
{
|
|
applet->pause_scanning_item = gtk_image_menu_item_new_with_label (_("Pause Wireless Scanning"));
|
|
image = gtk_image_new_from_stock (GTK_STOCK_MEDIA_PAUSE, GTK_ICON_SIZE_MENU);
|
|
}
|
|
else
|
|
{
|
|
applet->pause_scanning_item = gtk_image_menu_item_new_with_label (_("Resume Wireless Scanning"));
|
|
image = gtk_image_new_from_stock (GTK_STOCK_MEDIA_PLAY, GTK_ICON_SIZE_MENU);
|
|
}
|
|
g_signal_connect (G_OBJECT (applet->pause_scanning_item), "activate", G_CALLBACK (nmwa_set_scanning_enabled_cb), applet);
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (applet->pause_scanning_item), image);
|
|
gtk_menu_shell_insert (GTK_MENU_SHELL (applet->context_menu), applet->pause_scanning_item, 0);
|
|
gtk_widget_show_all (applet->pause_scanning_item);
|
|
|
|
if (applet->wireless_enabled)
|
|
{
|
|
applet->stop_wireless_item = gtk_image_menu_item_new_with_label (_("Stop All Wireless Devices"));
|
|
image = gtk_image_new_from_stock (GTK_STOCK_STOP, GTK_ICON_SIZE_MENU);
|
|
}
|
|
else
|
|
{
|
|
applet->stop_wireless_item = gtk_image_menu_item_new_with_label (_("Start All Wireless Devices"));
|
|
image = gtk_image_new_from_stock (GTK_STOCK_MEDIA_PLAY, GTK_ICON_SIZE_MENU);
|
|
}
|
|
g_signal_connect (G_OBJECT (applet->stop_wireless_item), "activate", G_CALLBACK (nmwa_set_wireless_enabled_cb), applet);
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (applet->stop_wireless_item), image);
|
|
gtk_menu_shell_insert (GTK_MENU_SHELL (applet->context_menu), applet->stop_wireless_item, 1);
|
|
gtk_widget_show_all (applet->stop_wireless_item);
|
|
|
|
g_mutex_unlock (applet->data_mutex);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_context_menu_create
|
|
*
|
|
* Generate the contextual popup menu.
|
|
*
|
|
*/
|
|
static GtkWidget *nmwa_context_menu_create (NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget *menu;
|
|
GtkWidget *menu_item;
|
|
GtkWidget *image;
|
|
|
|
g_return_val_if_fail (applet != NULL, NULL);
|
|
|
|
menu = gtk_menu_new ();
|
|
|
|
applet->pause_scanning_item = gtk_image_menu_item_new_with_label (_("Pause Wireless Scanning"));
|
|
g_signal_connect (G_OBJECT (applet->pause_scanning_item), "activate", G_CALLBACK (nmwa_set_scanning_enabled_cb), applet);
|
|
image = gtk_image_new_from_stock (GTK_STOCK_MEDIA_PAUSE, GTK_ICON_SIZE_MENU);
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (applet->pause_scanning_item), image);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), applet->pause_scanning_item);
|
|
|
|
applet->stop_wireless_item = gtk_image_menu_item_new_with_label (_("Stop All Wireless Devices"));
|
|
g_signal_connect (G_OBJECT (applet->stop_wireless_item), "activate", G_CALLBACK (nmwa_set_wireless_enabled_cb), applet);
|
|
image = gtk_image_new_from_stock (GTK_STOCK_STOP, GTK_ICON_SIZE_MENU);
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (applet->stop_wireless_item), image);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), applet->stop_wireless_item);
|
|
|
|
menu_item = gtk_separator_menu_item_new ();
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
|
|
menu_item = gtk_image_menu_item_new_with_label (_("Help"));
|
|
/* g_signal_connect (G_OBJECT (menu_item), "activate", G_CALLBACK (nmwa_help_cb), applet); */
|
|
image = gtk_image_new_from_stock (GTK_STOCK_HELP, GTK_ICON_SIZE_MENU);
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menu_item), image);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
gtk_widget_set_sensitive (GTK_WIDGET (menu_item), FALSE);
|
|
|
|
menu_item = gtk_image_menu_item_new_with_label (_("About"));
|
|
g_signal_connect (G_OBJECT (menu_item), "activate", G_CALLBACK (nmwa_about_cb), applet);
|
|
image = gtk_image_new_from_stock (GTK_STOCK_ABOUT, GTK_ICON_SIZE_MENU);
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menu_item), image);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
|
|
|
|
gtk_widget_show_all (menu);
|
|
|
|
return menu;
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_theme_change_cb
|
|
*
|
|
* Destroy the popdown menu when the theme changes
|
|
*
|
|
*/
|
|
static void nmwa_theme_change_cb (NMWirelessApplet *applet)
|
|
{
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if (applet->dropdown_menu)
|
|
nmwa_dropdown_menu_clear (applet->dropdown_menu);
|
|
|
|
if (applet->top_menu_item)
|
|
{
|
|
gtk_menu_item_remove_submenu (GTK_MENU_ITEM (applet->top_menu_item));
|
|
applet->dropdown_menu = nmwa_dropdown_menu_create (GTK_MENU_ITEM (applet->top_menu_item), applet);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* nmwa_toplevel_menu_button_press_cb
|
|
*
|
|
* Handle right-clicks for the context popup menu
|
|
*
|
|
*/
|
|
static gboolean nmwa_toplevel_menu_button_press_cb (GtkWidget *widget, GdkEventButton *event, gpointer user_data)
|
|
{
|
|
NMWirelessApplet *applet = (NMWirelessApplet *)user_data;
|
|
|
|
g_return_val_if_fail (applet != NULL, FALSE);
|
|
|
|
if (event->button != 1)
|
|
g_signal_stop_emission_by_name (widget, "button_press_event");
|
|
|
|
if (event->button == 3)
|
|
{
|
|
nmwa_context_menu_update (applet);
|
|
gtk_menu_popup (GTK_MENU (applet->context_menu), NULL, NULL, NULL, applet, event->button, event->time);
|
|
return (TRUE);
|
|
}
|
|
|
|
return (FALSE);
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_setup_widgets
|
|
*
|
|
* Intialize the applet's widgets and packing, create the initial
|
|
* menu of networks.
|
|
*
|
|
*/
|
|
static void nmwa_setup_widgets (NMWirelessApplet *applet)
|
|
{
|
|
GtkWidget *menu_bar;
|
|
GtkWidget *event_box;
|
|
|
|
/* Event box for tooltips */
|
|
applet->event_box = gtk_event_box_new ();
|
|
gtk_container_set_border_width (GTK_CONTAINER (applet->event_box), 0);
|
|
|
|
menu_bar = gtk_menu_bar_new ();
|
|
|
|
applet->top_menu_item = gtk_menu_item_new();
|
|
gtk_widget_set_name (applet->top_menu_item, "ToplevelMenu");
|
|
gtk_container_set_border_width (GTK_CONTAINER (applet->top_menu_item), 0);
|
|
g_signal_connect (applet->top_menu_item, "button_press_event", G_CALLBACK (nmwa_toplevel_menu_button_press_cb), applet);
|
|
|
|
applet->dropdown_menu = nmwa_dropdown_menu_create (GTK_MENU_ITEM (applet->top_menu_item), applet);
|
|
|
|
applet->pixmap = gtk_image_new ();
|
|
|
|
/* Set up the widget structure and show the applet */
|
|
gtk_container_add (GTK_CONTAINER(applet->top_menu_item), applet->pixmap);
|
|
gtk_menu_shell_append (GTK_MENU_SHELL(menu_bar), applet->top_menu_item);
|
|
gtk_container_add (GTK_CONTAINER(applet->event_box), menu_bar);
|
|
gtk_container_add (GTK_CONTAINER (applet), applet->event_box);
|
|
gtk_widget_show_all (GTK_WIDGET (applet));
|
|
|
|
applet->context_menu = nmwa_context_menu_create (applet);
|
|
applet->encryption_size_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
|
|
}
|
|
|
|
/*
|
|
* nmwa_gconf_networks_notify_callback
|
|
*
|
|
* Callback from gconf when wireless networking key/values have changed.
|
|
*
|
|
*/
|
|
static void nmwa_gconf_networks_notify_callback (GConfClient *client, guint connection_id, GConfEntry *entry, gpointer user_data)
|
|
{
|
|
NMWirelessApplet * applet = (NMWirelessApplet *)user_data;
|
|
const char * key = NULL;
|
|
|
|
g_return_if_fail (client != NULL);
|
|
g_return_if_fail (entry != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if ((key = gconf_entry_get_key (entry)))
|
|
{
|
|
int path_len = strlen (GCONF_PATH_WIRELESS_NETWORKS) + 1;
|
|
|
|
if (strncmp (GCONF_PATH_WIRELESS_NETWORKS"/", key, path_len) == 0)
|
|
{
|
|
char *network = g_strdup ((key + path_len));
|
|
char *slash_pos;
|
|
char *unescaped_network;
|
|
|
|
/* If its a key under the network name, zero out the slash so we
|
|
* are left with only the network name.
|
|
*/
|
|
unescaped_network = gconf_unescape_key (network, strlen (network));
|
|
if ((slash_pos = strchr (unescaped_network, '/')))
|
|
*slash_pos = '\0';
|
|
|
|
// nmi_dbus_signal_update_network (applet->connection, unescaped_network, NETWORK_TYPE_ALLOWED);
|
|
g_free (unescaped_network);
|
|
g_free (network);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_gconf_vpn_connections_notify_callback
|
|
*
|
|
* Callback from gconf when VPN connection values have changed.
|
|
*
|
|
*/
|
|
static void nmwa_gconf_vpn_connections_notify_callback (GConfClient *client, guint connection_id, GConfEntry *entry, gpointer user_data)
|
|
{
|
|
NMWirelessApplet * applet = (NMWirelessApplet *)user_data;
|
|
const char * key = NULL;
|
|
|
|
g_return_if_fail (client != NULL);
|
|
g_return_if_fail (entry != NULL);
|
|
g_return_if_fail (applet != NULL);
|
|
|
|
if ((key = gconf_entry_get_key (entry)))
|
|
{
|
|
int path_len = strlen (GCONF_PATH_VPN_CONNECTIONS) + 1;
|
|
|
|
if (strncmp (GCONF_PATH_VPN_CONNECTIONS"/", key, path_len) == 0)
|
|
{
|
|
char *name = g_strdup ((key + path_len));
|
|
char *slash_pos;
|
|
char *unescaped_name;
|
|
|
|
/* If its a key under the the VPN name, zero out the slash so we
|
|
* are left with only the VPN name.
|
|
*/
|
|
unescaped_name = gconf_unescape_key (name, strlen (name));
|
|
if ((slash_pos = strchr (unescaped_name, '/')))
|
|
*slash_pos = '\0';
|
|
|
|
// nmi_dbus_signal_update_vpn_connection (info->connection, unescaped_name);
|
|
g_free (unescaped_name);
|
|
g_free (name);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_destroy
|
|
*
|
|
* Destroy the applet and clean up its data
|
|
*
|
|
*/
|
|
static void nmwa_destroy (NMWirelessApplet *applet, gpointer user_data)
|
|
{
|
|
if (applet->dropdown_menu)
|
|
nmwa_dropdown_menu_clear (applet->dropdown_menu);
|
|
if (applet->top_menu_item)
|
|
gtk_menu_item_remove_submenu (GTK_MENU_ITEM (applet->top_menu_item));
|
|
|
|
nmwa_icons_free (applet);
|
|
|
|
nmi_passphrase_dialog_destroy (applet->passphrase_dialog);
|
|
|
|
if (applet->redraw_timeout_id > 0)
|
|
{
|
|
gtk_timeout_remove (applet->redraw_timeout_id);
|
|
applet->redraw_timeout_id = 0;
|
|
}
|
|
|
|
g_main_loop_quit (applet->thread_loop);
|
|
g_thread_join (applet->dbus_thread);
|
|
|
|
if (applet->gconf_client)
|
|
g_object_unref (G_OBJECT (applet->gconf_client));
|
|
|
|
nmwa_free_gui_data_model (applet);
|
|
nmwa_free_dbus_data_model (applet);
|
|
|
|
g_free (applet->glade_file);
|
|
|
|
gconf_client_notify_remove (applet->gconf_client, applet->gconf_net_notify_id);
|
|
gconf_client_notify_remove (applet->gconf_client, applet->gconf_vpn_notify_id);
|
|
g_object_unref (G_OBJECT (applet->gconf_client));
|
|
}
|
|
|
|
|
|
/*
|
|
* nmwa_get_instance
|
|
*
|
|
* Create the initial instance of our wireless applet
|
|
*
|
|
*/
|
|
static GtkWidget * nmwa_get_instance (NMWirelessApplet *applet)
|
|
{
|
|
GError * error = NULL;
|
|
|
|
gtk_widget_hide (GTK_WIDGET (applet));
|
|
|
|
applet->nm_running = FALSE;
|
|
applet->dev_pending_call_list = NULL;
|
|
applet->dbus_device_list = NULL;
|
|
applet->dbus_active_device = NULL;
|
|
applet->dbus_active_device_path = NULL;
|
|
applet->dbus_active_vpn_name = NULL;
|
|
applet->dbus_vpn_connections = NULL;
|
|
applet->dbus_nm_state = NM_STATE_DISCONNECTED;
|
|
applet->vpn_pending_call_list = NULL;
|
|
applet->gui_device_list = NULL;
|
|
applet->gui_active_device = NULL;
|
|
applet->gui_active_vpn = NULL;
|
|
applet->gui_vpn_connections = NULL;
|
|
applet->gui_nm_state = NM_STATE_DISCONNECTED;
|
|
applet->tooltips = NULL;
|
|
applet->thread_context = NULL;
|
|
applet->thread_loop = NULL;
|
|
applet->thread_done = FALSE;
|
|
|
|
applet->glade_file = g_build_filename (GLADEDIR, "wireless-applet.glade", NULL);
|
|
if (!applet->glade_file || !g_file_test (applet->glade_file, G_FILE_TEST_IS_REGULAR))
|
|
{
|
|
show_warning_dialog (_("The NetworkManager Applet could not find some required resources (the glade file was not found)."));
|
|
g_free (applet->glade_file);
|
|
applet->glade_file = NULL;
|
|
return NULL;
|
|
}
|
|
|
|
applet->passphrase_dialog = nmi_passphrase_dialog_init (applet);
|
|
|
|
applet->gconf_client = gconf_client_get_default ();
|
|
if (!applet->gconf_client)
|
|
return NULL;
|
|
|
|
gconf_client_add_dir (applet->gconf_client, GCONF_PATH_WIRELESS_NETWORKS, GCONF_CLIENT_PRELOAD_NONE, NULL);
|
|
applet->gconf_net_notify_id = gconf_client_notify_add (applet->gconf_client, GCONF_PATH_WIRELESS_NETWORKS,
|
|
nmwa_gconf_networks_notify_callback, applet, NULL, NULL);
|
|
|
|
gconf_client_add_dir (applet->gconf_client, GCONF_PATH_VPN_CONNECTIONS, GCONF_CLIENT_PRELOAD_NONE, NULL);
|
|
applet->gconf_vpn_notify_id = gconf_client_notify_add (applet->gconf_client, GCONF_PATH_VPN_CONNECTIONS,
|
|
nmwa_gconf_vpn_connections_notify_callback, applet, NULL, NULL);
|
|
|
|
/* Start our dbus thread */
|
|
if (!(applet->data_mutex = g_mutex_new ()))
|
|
{
|
|
g_object_unref (G_OBJECT (applet->gconf_client));
|
|
return NULL;
|
|
}
|
|
if (!(applet->dbus_thread = g_thread_create (nmwa_dbus_worker, applet, FALSE, &error)))
|
|
{
|
|
g_mutex_free (applet->data_mutex);
|
|
g_object_unref (G_OBJECT (applet->gconf_client));
|
|
return NULL;
|
|
}
|
|
|
|
/* Load pixmaps and create applet widgets */
|
|
nmwa_setup_widgets (applet);
|
|
|
|
g_signal_connect (applet, "destroy", G_CALLBACK (nmwa_destroy), NULL);
|
|
g_signal_connect (applet, "style-set", G_CALLBACK (nmwa_theme_change_cb), NULL);
|
|
|
|
/* Start redraw timeout */
|
|
nmwa_start_redraw_timeout (applet);
|
|
|
|
return GTK_WIDGET (applet);
|
|
}
|
|
|
|
static void setup_stock (void)
|
|
{
|
|
GtkIconFactory *ifactory;
|
|
GtkIconSet *iset;
|
|
GtkIconSource *isource;
|
|
static gboolean initted = FALSE;
|
|
|
|
if (initted)
|
|
return;
|
|
|
|
ifactory = gtk_icon_factory_new ();
|
|
iset = gtk_icon_set_new ();
|
|
isource = gtk_icon_source_new ();
|
|
|
|
/* we use the lockscreen icon to get a key */
|
|
gtk_icon_source_set_icon_name (isource, "gnome-lockscreen");
|
|
gtk_icon_set_add_source (iset, isource);
|
|
gtk_icon_factory_add (ifactory, "gnome-lockscreen", iset);
|
|
gtk_icon_factory_add_default (ifactory);
|
|
|
|
initted = TRUE;
|
|
}
|
|
|
|
static void nmwa_icons_free (NMWirelessApplet *applet)
|
|
{
|
|
gint i;
|
|
|
|
g_object_unref (applet->no_connection_icon);
|
|
g_object_unref (applet->wired_icon);
|
|
g_object_unref (applet->adhoc_icon);
|
|
g_object_unref (applet->vpn_lock_icon);
|
|
|
|
g_object_unref (applet->wireless_00_icon);
|
|
g_object_unref (applet->wireless_25_icon);
|
|
g_object_unref (applet->wireless_50_icon);
|
|
g_object_unref (applet->wireless_75_icon);
|
|
g_object_unref (applet->wireless_100_icon);
|
|
|
|
for (i = 0; i < NUM_WIRED_CONNECTING_FRAMES; i++)
|
|
g_object_unref (applet->wired_connecting_icons[i]);
|
|
|
|
for (i = 0; i < NUM_WIRELESS_CONNECTING_FRAMES; i++)
|
|
g_object_unref (applet->wireless_connecting_icons[i]);
|
|
|
|
for (i = 0; i < NUM_WIRELESS_SCANNING_FRAMES; i++)
|
|
g_object_unref (applet->wireless_scanning_icons[i]);
|
|
}
|
|
|
|
static void
|
|
nmwa_icons_load_from_disk (NMWirelessApplet *applet, GtkIconTheme *icon_theme)
|
|
{
|
|
char * name;
|
|
int i;
|
|
gboolean success = TRUE;
|
|
|
|
/* Assume icons are square */
|
|
gint icon_size = 22;
|
|
|
|
applet->no_connection_icon = gtk_icon_theme_load_icon (icon_theme, "nm-no-connection", icon_size, 0, NULL);
|
|
applet->wired_icon = gtk_icon_theme_load_icon (icon_theme, "nm-device-wired", icon_size, 0, NULL);
|
|
applet->adhoc_icon = gtk_icon_theme_load_icon (icon_theme, "nm-adhoc", icon_size, 0, NULL);
|
|
applet->vpn_lock_icon = gtk_icon_theme_load_icon (icon_theme, "nm-vpn-lock", icon_size, 0, NULL);
|
|
|
|
applet->wireless_00_icon = gtk_icon_theme_load_icon (icon_theme, "nm-signal-00", icon_size, 0, NULL);
|
|
applet->wireless_25_icon = gtk_icon_theme_load_icon (icon_theme, "nm-signal-25", icon_size, 0, NULL);
|
|
applet->wireless_50_icon = gtk_icon_theme_load_icon (icon_theme, "nm-signal-50", icon_size, 0, NULL);
|
|
applet->wireless_75_icon = gtk_icon_theme_load_icon (icon_theme, "nm-signal-75", icon_size, 0, NULL);
|
|
applet->wireless_100_icon = gtk_icon_theme_load_icon (icon_theme, "nm-signal-100", icon_size, 0, NULL);
|
|
|
|
if (!applet->no_connection_icon || !applet->wired_icon || !applet->adhoc_icon || !applet->vpn_lock_icon
|
|
|| !applet->wireless_00_icon || !applet->wireless_25_icon || !applet->wireless_50_icon || !applet->wireless_75_icon
|
|
|| !applet->wireless_100_icon)
|
|
success = FALSE;
|
|
|
|
for (i = 0; i < NUM_WIRED_CONNECTING_FRAMES; i++)
|
|
{
|
|
name = g_strdup_printf ("nm-connecting%02d", i+1);
|
|
applet->wired_connecting_icons[i] = gtk_icon_theme_load_icon (icon_theme, name, icon_size, 0, NULL);
|
|
g_free (name);
|
|
if (!applet->wired_connecting_icons[i])
|
|
success = FALSE;
|
|
}
|
|
|
|
for (i = 0; i < NUM_WIRELESS_CONNECTING_FRAMES; i++)
|
|
{
|
|
name = g_strdup_printf ("nm-connecting%02d", i+1);
|
|
applet->wireless_connecting_icons[i] = gtk_icon_theme_load_icon (icon_theme, name, icon_size, 0, NULL);
|
|
g_free (name);
|
|
if (!applet->wireless_connecting_icons[i])
|
|
success = FALSE;
|
|
}
|
|
|
|
for (i = 0; i < NUM_WIRELESS_SCANNING_FRAMES; i++)
|
|
{
|
|
name = g_strdup_printf ("nm-detect%02d", i+1);
|
|
applet->wireless_scanning_icons[i] = gtk_icon_theme_load_icon (icon_theme, name, icon_size, 0, NULL);
|
|
g_free (name);
|
|
if (!applet->wireless_scanning_icons[i])
|
|
success = FALSE;
|
|
}
|
|
|
|
if (!success)
|
|
{
|
|
show_warning_dialog (_("The NetworkManager applet could not find some required resources. It cannot continue.\n"));
|
|
exit (1);
|
|
}
|
|
}
|
|
|
|
static void nmwa_icon_theme_changed (GtkIconTheme *icon_theme, NMWirelessApplet *applet)
|
|
{
|
|
nmwa_icons_free (applet);
|
|
nmwa_icons_load_from_disk (applet, icon_theme);
|
|
/* FIXME: force redraw */
|
|
}
|
|
|
|
const gchar *style = " \
|
|
style \"MenuBar\" \
|
|
{ \
|
|
GtkMenuBar::shadow_type = GTK_SHADOW_NONE \
|
|
GtkMenuBar::internal-padding = 0 \
|
|
} \
|
|
style \"MenuItem\" \
|
|
{ \
|
|
xthickness=0 \
|
|
ythickness=0 \
|
|
} \
|
|
class \"GtkMenuBar\" style \"MenuBar\"\
|
|
widget \"*ToplevelMenu*\" style \"MenuItem\"\
|
|
";
|
|
|
|
static void nmwa_icons_init (NMWirelessApplet *applet)
|
|
{
|
|
GtkIconTheme *icon_theme;
|
|
|
|
/* FIXME: Do we need to worry about other screens? */
|
|
gtk_rc_parse_string (style);
|
|
|
|
icon_theme = gtk_icon_theme_get_default ();
|
|
nmwa_icons_load_from_disk (applet, icon_theme);
|
|
g_signal_connect (icon_theme, "changed", G_CALLBACK (nmwa_icon_theme_changed), applet);
|
|
}
|
|
|
|
|
|
NMWirelessApplet *nmwa_new ()
|
|
{
|
|
return g_object_new (NM_TYPE_WIRELESS_APPLET, "title", "NetworkManager", NULL);
|
|
}
|
|
|