/* * * Copyright (C) 2013 Bastien Nocera * * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include "cc-bluetooth-panel.h" #include "cc-bluetooth-resources.h" #define WID(s) GTK_WIDGET (gtk_builder_get_object (self->builder, s)) #define BLUETOOTH_NO_DEVICES_PAGE "no-devices-page" #define BLUETOOTH_DISABLED_PAGE "disabled-page" #define BLUETOOTH_AIRPLANE_PAGE "airplane-page" #define BLUETOOTH_HW_AIRPLANE_PAGE "hw-airplane-page" #define BLUETOOTH_WORKING_PAGE "working-page" struct _CcBluetoothPanel { CcPanel parent_instance; GtkBuilder *builder; GtkWidget *stack; GtkWidget *widget; GCancellable *cancellable; /* Killswitch */ GtkWidget *kill_switch_header; GDBusProxy *rfkill, *properties; gboolean airplane_mode; gboolean bt_airplane_mode; gboolean hardware_airplane_mode; gboolean has_airplane_mode; }; CC_PANEL_REGISTER (CcBluetoothPanel, cc_bluetooth_panel) static void cc_bluetooth_panel_finalize (GObject *object); static void cc_bluetooth_panel_constructed (GObject *object); static const char * cc_bluetooth_panel_get_help_uri (CcPanel *panel) { return "help:gnome-help/bluetooth"; } static void cc_bluetooth_panel_class_init (CcBluetoothPanelClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); CcPanelClass *panel_class = CC_PANEL_CLASS (klass); object_class->constructed = cc_bluetooth_panel_constructed; object_class->finalize = cc_bluetooth_panel_finalize; panel_class->get_help_uri = cc_bluetooth_panel_get_help_uri; } static void cc_bluetooth_panel_finalize (GObject *object) { CcBluetoothPanel *self; self = CC_BLUETOOTH_PANEL (object); g_cancellable_cancel (self->cancellable); g_clear_object (&self->cancellable); g_clear_object (&self->properties); g_clear_object (&self->rfkill); g_clear_object (&self->kill_switch_header); G_OBJECT_CLASS (cc_bluetooth_panel_parent_class)->finalize (object); } static void cc_bluetooth_panel_constructed (GObject *object) { CcBluetoothPanel *self = CC_BLUETOOTH_PANEL (object); G_OBJECT_CLASS (cc_bluetooth_panel_parent_class)->constructed (object); /* add kill switch widgets */ self->kill_switch_header = g_object_ref (WID ("box_power")); cc_shell_embed_widget_in_header (cc_panel_get_shell (CC_PANEL (self)), self->kill_switch_header); gtk_widget_show (self->kill_switch_header); } static void power_callback (GObject *object, GParamSpec *spec, CcBluetoothPanel *self) { gboolean state; state = gtk_switch_get_active (GTK_SWITCH (WID ("switch_bluetooth"))); g_debug ("Power switched to %s", state ? "on" : "off"); g_dbus_proxy_call (self->properties, "Set", g_variant_new_parsed ("('org.gnome.SettingsDaemon.Rfkill', 'BluetoothAirplaneMode', %v)", g_variant_new_boolean (!state)), G_DBUS_CALL_FLAGS_NONE, -1, self->cancellable, NULL, NULL); } static void cc_bluetooth_panel_update_power (CcBluetoothPanel *self) { GtkAlign valign; GObject *toggle; gboolean sensitive, powered, change_powered; const char *page; g_debug ("Updating airplane mode: BluetoothHasAirplaneMode %d, BluetoothHardwareAirplaneMode %d, BluetoothAirplaneMode %d, AirplaneMode %d", self->has_airplane_mode, self->hardware_airplane_mode, self->bt_airplane_mode, self->airplane_mode); change_powered = TRUE; valign = GTK_ALIGN_CENTER; if (self->has_airplane_mode == FALSE) { g_debug ("No Bluetooth available"); sensitive = FALSE; powered = FALSE; page = BLUETOOTH_NO_DEVICES_PAGE; } else if (self->hardware_airplane_mode) { g_debug ("Bluetooth is Hard blocked"); sensitive = FALSE; powered = FALSE; page = BLUETOOTH_HW_AIRPLANE_PAGE; } else if (self->airplane_mode) { g_debug ("Airplane mode is on, Wi-Fi and Bluetooth are disabled"); sensitive = FALSE; powered = FALSE; page = BLUETOOTH_AIRPLANE_PAGE; } else if (self->bt_airplane_mode || !bluetooth_settings_widget_get_default_adapter_powered (BLUETOOTH_SETTINGS_WIDGET (self->widget))) { g_debug ("Default adapter is unpowered, but should be available"); sensitive = TRUE; change_powered = FALSE; page = BLUETOOTH_DISABLED_PAGE; } else { g_debug ("Bluetooth is available and powered"); sensitive = TRUE; powered = TRUE; page = BLUETOOTH_WORKING_PAGE; valign = GTK_ALIGN_FILL; } gtk_widget_set_valign (self->stack, valign); gtk_widget_set_sensitive (WID ("box_power") , sensitive); toggle = G_OBJECT (WID ("switch_bluetooth")); if (change_powered) { g_signal_handlers_block_by_func (toggle, power_callback, self); gtk_switch_set_active (GTK_SWITCH (toggle), powered); g_signal_handlers_unblock_by_func (toggle, power_callback, self); } gtk_stack_set_visible_child_name (GTK_STACK (self->stack), page); } static void airplane_mode_changed (GDBusProxy *proxy, GVariant *changed_properties, GStrv invalidated_properties, CcBluetoothPanel *self) { GVariant *v; v = g_dbus_proxy_get_cached_property (self->rfkill, "AirplaneMode"); self->airplane_mode = g_variant_get_boolean (v); g_variant_unref (v); v = g_dbus_proxy_get_cached_property (self->rfkill, "BluetoothAirplaneMode"); self->bt_airplane_mode = g_variant_get_boolean (v); g_variant_unref (v); v = g_dbus_proxy_get_cached_property (self->rfkill, "BluetoothHardwareAirplaneMode"); self->hardware_airplane_mode = g_variant_get_boolean (v); g_variant_unref (v); v = g_dbus_proxy_get_cached_property (self->rfkill, "BluetoothHasAirplaneMode"); self->has_airplane_mode = g_variant_get_boolean (v); g_variant_unref (v); cc_bluetooth_panel_update_power (self); } static void on_airplane_mode_off_clicked (GtkButton *button, CcBluetoothPanel *self) { g_debug ("Airplane Mode Off clicked, disabling airplane mode"); g_dbus_proxy_call (self->rfkill, "org.freedesktop.DBus.Properties.Set", g_variant_new_parsed ("('org.gnome.SettingsDaemon.Rfkill'," "'AirplaneMode', %v)", g_variant_new_boolean (FALSE)), G_DBUS_CALL_FLAGS_NONE, -1, self->cancellable, NULL, NULL); } static void add_stack_page (CcBluetoothPanel *self, const char *message, const char *explanation, const char *name) { GtkWidget *label, *image, *box; char *str; box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); gtk_widget_show (box); g_object_set (G_OBJECT (box), "margin-top", 64, "margin-bottom", 64, NULL); g_object_set (G_OBJECT (box), "margin-start", 12, "margin-end", 12, NULL); if (g_str_equal (name, BLUETOOTH_AIRPLANE_PAGE) || g_str_equal (name, BLUETOOTH_HW_AIRPLANE_PAGE)) { image = gtk_image_new_from_icon_name ("airplane-mode-symbolic", GTK_ICON_SIZE_DIALOG); } else { image = gtk_image_new_from_icon_name ("bluetooth-active-symbolic", GTK_ICON_SIZE_DIALOG); } gtk_widget_show (image); gtk_image_set_pixel_size (GTK_IMAGE (image), 192); gtk_style_context_add_class (gtk_widget_get_style_context (image), "dim-label"); gtk_box_pack_start (GTK_BOX (box), image, FALSE, FALSE, 24); str = g_strdup_printf ("%s", message); label = gtk_label_new (""); gtk_widget_show (label); gtk_label_set_markup (GTK_LABEL (label), str); g_free (str); gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 0); label = gtk_label_new (explanation); gtk_widget_show (label); gtk_label_set_line_wrap (GTK_LABEL (label), TRUE); gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 0); if (g_str_equal (name, BLUETOOTH_AIRPLANE_PAGE)) { GtkWidget *button; button = gtk_button_new_with_label (_("Turn Off Airplane Mode")); gtk_widget_show (button); gtk_widget_set_valign (button, GTK_ALIGN_CENTER); gtk_widget_set_halign (button, GTK_ALIGN_CENTER); g_signal_connect (G_OBJECT (button), "clicked", G_CALLBACK (on_airplane_mode_off_clicked), self); gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 24); } gtk_stack_add_named (GTK_STACK (self->stack), box, name); } static void panel_changed (GtkWidget *settings_widget, const char *panel, CcBluetoothPanel *self) { CcShell *shell; GError *error = NULL; shell = cc_panel_get_shell (CC_PANEL (self)); if (cc_shell_set_active_panel_from_id (shell, panel, NULL, &error) == FALSE) { g_warning ("Failed to activate '%s' panel: %s", panel, error->message); g_error_free (error); } } static void cc_bluetooth_panel_init (CcBluetoothPanel *self) { GError *error = NULL; g_resources_register (cc_bluetooth_get_resource ()); self->builder = gtk_builder_new (); gtk_builder_set_translation_domain (self->builder, GETTEXT_PACKAGE); gtk_builder_add_from_resource (self->builder, "/org/gnome/control-center/bluetooth/bluetooth.ui", &error); if (error != NULL) { g_warning ("Could not load ui: %s", error->message); g_error_free (error); return; } self->cancellable = g_cancellable_new (); /* RFKill */ self->rfkill = cc_object_storage_create_dbus_proxy_sync (G_BUS_TYPE_SESSION, G_DBUS_PROXY_FLAGS_NONE, "org.gnome.SettingsDaemon.Rfkill", "/org/gnome/SettingsDaemon/Rfkill", "org.gnome.SettingsDaemon.Rfkill", NULL, NULL); self->properties = cc_object_storage_create_dbus_proxy_sync (G_BUS_TYPE_SESSION, G_DBUS_PROXY_FLAGS_NONE, "org.gnome.SettingsDaemon.Rfkill", "/org/gnome/SettingsDaemon/Rfkill", "org.freedesktop.DBus.Properties", NULL, NULL); self->stack = gtk_stack_new (); gtk_stack_set_homogeneous (GTK_STACK (self->stack), TRUE); add_stack_page (self, _("No Bluetooth Found"), _("Plug in a dongle to use Bluetooth."), BLUETOOTH_NO_DEVICES_PAGE); add_stack_page (self, _("Bluetooth Turned Off"), _("Turn on to connect devices and receive file transfers."), BLUETOOTH_DISABLED_PAGE); add_stack_page (self, _("Airplane Mode is on"), _("Bluetooth is disabled when airplane mode is on."), BLUETOOTH_AIRPLANE_PAGE); add_stack_page (self, _("Hardware Airplane Mode is on"), _("Turn off the Airplane mode switch to enable Bluetooth."), BLUETOOTH_HW_AIRPLANE_PAGE); self->widget = bluetooth_settings_widget_new (); g_signal_connect (G_OBJECT (self->widget), "panel-changed", G_CALLBACK (panel_changed), self); gtk_stack_add_named (GTK_STACK (self->stack), self->widget, BLUETOOTH_WORKING_PAGE); gtk_widget_show (self->widget); gtk_widget_show (self->stack); gtk_container_add (GTK_CONTAINER (self), self->stack); airplane_mode_changed (NULL, NULL, NULL, self); g_signal_connect_object (self->rfkill, "g-properties-changed", G_CALLBACK (airplane_mode_changed), self, 0); g_signal_connect_object (G_OBJECT (self->widget), "adapter-status-changed", G_CALLBACK (cc_bluetooth_panel_update_power), self, G_CONNECT_SWAPPED); g_signal_connect (G_OBJECT (WID ("switch_bluetooth")), "notify::active", G_CALLBACK (power_callback), self); }