/* * Copyright 2013 Canonical Ltd. * * This program is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License version 3, as published * by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranties of * MERCHANTABILITY, SATISFACTORY QUALITY, 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, see <http://www.gnu.org/licenses/>. * * Authors: * Charles Kerr <charles.kerr@canonical.com> */ #include <datetime/dbus-shared.h> #include <datetime/timezone-timedated.h> #include <gio/gio.h> #include <cerrno> #include <cstdlib> namespace ayatana { namespace indicator { namespace datetime { /*** **** ***/ class TimedatedTimezone::Impl { public: Impl(TimedatedTimezone& owner, GDBusConnection* connection): m_owner{owner}, m_connection{G_DBUS_CONNECTION(g_object_ref(G_OBJECT(connection)))}, m_cancellable{g_cancellable_new()} { // set the fallback value m_owner.timezone.set("Etc/Utc"); // watch for timedate1 on the bus m_watcher_id = g_bus_watch_name_on_connection( m_connection, Bus::Timedate1::BUSNAME, G_BUS_NAME_WATCHER_FLAGS_AUTO_START, on_timedate1_appeared, on_timedate1_vanished, this, nullptr); // listen for changed properties m_signal_subscription_id = g_dbus_connection_signal_subscribe( m_connection, Bus::Timedate1::IFACE, Bus::Properties::IFACE, Bus::Properties::Signals::PROPERTIES_CHANGED, Bus::Timedate1::ADDR, nullptr, G_DBUS_SIGNAL_FLAGS_NONE, on_properties_changed, this, nullptr); } ~Impl() { g_cancellable_cancel(m_cancellable); g_clear_object(&m_cancellable); g_bus_unwatch_name(m_watcher_id); g_dbus_connection_signal_unsubscribe(m_connection, m_signal_subscription_id); g_clear_object(&m_connection); } private: static void on_timedate1_appeared(GDBusConnection * /*connection*/, const gchar * name, const gchar * /*name_owner*/, gpointer gself) { g_debug("%s appeared on bus", name); static_cast<Impl*>(gself)->ask_for_timezone(); } static void on_timedate1_vanished(GDBusConnection * /*connection*/, const gchar * name, gpointer /*gself*/) { g_debug("%s not present on bus", name); } static void on_properties_changed(GDBusConnection * /*connection*/, const gchar * /*sender_name*/, const gchar * /*object_path*/, const gchar * /*interface_name*/, const gchar * /*signal_name*/, GVariant * parameters, gpointer gself) { auto self = static_cast<Impl*>(gself); GVariant* changed_properties {}; gchar** invalidated_properties {}; g_variant_get(parameters, "(s@a{sv}^as)", NULL, &changed_properties, &invalidated_properties); const char* tz {}; if (g_variant_lookup(changed_properties, Bus::Timedate1::Properties::TIMEZONE, "&s", &tz, NULL)) { if (tz != nullptr) self->set_timezone(tz); else g_warning("%s no timezone found", G_STRLOC); } else if (g_strv_contains(invalidated_properties, Bus::Timedate1::Properties::TIMEZONE)) { self->ask_for_timezone(); } g_variant_unref(changed_properties); g_strfreev(invalidated_properties); } void ask_for_timezone() { g_dbus_connection_call( m_connection, Bus::Timedate1::BUSNAME, Bus::Timedate1::ADDR, Bus::Properties::IFACE, Bus::Properties::Methods::GET, g_variant_new("(ss)", Bus::Timedate1::IFACE, Bus::Timedate1::Properties::TIMEZONE), G_VARIANT_TYPE("(v)"), G_DBUS_CALL_FLAGS_NONE, -1, m_cancellable, on_get_timezone_ready, this); } static void on_get_timezone_ready(GObject * connection, GAsyncResult * res, gpointer gself) { GError* error {}; GVariant* v = g_dbus_connection_call_finish(G_DBUS_CONNECTION(connection), res, &error); if (error != nullptr) { if (!g_error_matches(error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) g_warning("%s Couldn't get timezone: %s", G_STRLOC, error->message); } else if (v != nullptr) { GVariant* tzv {}; g_variant_get(v, "(v)", &tzv); const char* tz = g_variant_get_string(tzv, nullptr); if (tz != nullptr) static_cast<Impl*>(gself)->set_timezone(tz); else g_warning("%s no timezone found", G_STRLOC); g_clear_pointer(&tzv, g_variant_unref); g_clear_pointer(&v, g_variant_unref); } } void set_timezone(const std::string& tz) { g_return_if_fail(!tz.empty()); g_debug("set timezone: '%s'", tz.c_str()); m_owner.timezone.set(tz); } /*** **** ***/ TimedatedTimezone& m_owner; GDBusConnection* m_connection {}; GCancellable* m_cancellable {}; unsigned long m_signal_subscription_id {}; unsigned int m_watcher_id {}; }; /*** **** ***/ TimedatedTimezone::TimedatedTimezone(GDBusConnection* connection): impl{new Impl{*this, connection}} { } TimedatedTimezone::~TimedatedTimezone() { } /*** **** ***/ } // namespace datetime } // namespace indicator } // namespace ayatana