/* calls-application.c * * Copyright (C) 2018, 2019 Purism SPC * Copyright (C) 2018 Mohammed Sadiq * * This file is part of Calls. * * Calls 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 3 of the License, or * (at your option) any later version. * * Calls 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 Calls. If not, see . * * Authors: * Bob Ham * Mohammed Sadiq * * SPDX-License-Identifier: GPL-3.0-or-later */ #include "config.h" #include "calls-history-box.h" #include "calls-new-call-box.h" #include "calls-encryption-indicator.h" #include "calls-ringer.h" #include "calls-notifier.h" #include "calls-record-store.h" #include "calls-contacts.h" #include "calls-call-window.h" #include "calls-main-window.h" #include "calls-manager.h" #include "calls-application.h" #include #include #include #include /** * SECTION: calls-application * @title: CallsApplication * @short_description: Base Application class * @include: "calls-application.h" */ #define DEFAULT_PROVIDER_PLUGIN "mm" struct _CallsApplication { GtkApplication parent_instance; gboolean daemon; CallsManager *manager; CallsRinger *ringer; CallsNotifier *notifier; CallsRecordStore *record_store; CallsContacts *contacts; CallsMainWindow *main_window; CallsCallWindow *call_window; }; G_DEFINE_TYPE (CallsApplication, calls_application, GTK_TYPE_APPLICATION); static gboolean start_proper (CallsApplication *self); static gint handle_local_options (GApplication *application, GVariantDict *options) { gboolean ok; g_autoptr(GError) error = NULL; const gchar *arg; g_debug ("Registering application"); ok = g_application_register (application, NULL, &error); if (!ok) { g_error ("Error registering application: %s", error->message); } ok = g_variant_dict_lookup (options, "provider", "&s", &arg); if (ok) { g_action_group_activate_action (G_ACTION_GROUP (application), "set-provider-name", g_variant_new_string (arg)); } else { g_action_group_activate_action (G_ACTION_GROUP (application), "set-provider-name", g_variant_new_string (DEFAULT_PROVIDER_PLUGIN)); } ok = g_variant_dict_contains (options, "daemon"); if (ok) { g_action_group_activate_action (G_ACTION_GROUP (application), "set-daemon", NULL); } ok = g_variant_dict_lookup (options, "dial", "&s", &arg); if (ok) { g_action_group_activate_action (G_ACTION_GROUP (application), "dial", g_variant_new_string (arg)); } return -1; // Continue processing signal } static void set_provider_name_action (GSimpleAction *action, GVariant *parameter, gpointer user_data) { const gchar *name; name = g_variant_get_string (parameter, NULL); g_return_if_fail (name != NULL); /* FIXME: allow to set a new provider, we need to make sure that the provider is unloaded correctly from the CallsManager */ if (calls_manager_get_provider (calls_manager_get_default ()) != NULL) { g_warning ("Cannot set provider name to `%s'" " because provider is already created", name); return; } g_debug ("Start loading provider `%s'", name); calls_manager_set_provider (calls_manager_get_default (), name); } static void set_daemon_action (GSimpleAction *action, GVariant *parameter, gpointer user_data) { CallsApplication *self = CALLS_APPLICATION (user_data); if (self->main_window) { g_warning ("Cannot set application as a daemon" " because application is already started"); return; } self->daemon = TRUE; g_debug ("Application marked as daemon"); } #define DIALLING "0-9*#+ABCD" #define SIGNALLING ",TP!W@X" #define VISUAL "[:space:]\\-.()t/" #define REJECT_RE "[^" DIALLING SIGNALLING VISUAL "]" #define VISUAL_RE "[" VISUAL "]" static gboolean check_dial_number (const gchar *number) { GError *error = NULL; GRegex *reject; gboolean matches; reject = g_regex_new (REJECT_RE, 0, 0, &error); if (!reject) { g_warning ("Could not compile regex for" " dial number checking: %s", error->message); g_error_free (error); return FALSE; } matches = g_regex_match (reject, number, 0, NULL); g_regex_unref (reject); return !matches; } static gchar * extract_dial_string (const gchar *number) { g_autoptr(GError) error = NULL; g_autoptr(GRegex) replace_visual; gchar *dial_string; replace_visual = g_regex_new (VISUAL_RE, 0, 0, &error); if (!replace_visual) { g_warning ("Could not compile regex for" " dial number extracting: %s", error->message); return NULL; } dial_string = g_regex_replace_literal (replace_visual, number, -1, 0, "", 0, &error); if (!dial_string) { g_warning ("Error replacing visual separators" " in dial number: %s", error->message); return NULL; } return dial_string; } static void dial_action (GSimpleAction *action, GVariant *parameter, gpointer user_data) { CallsApplication *self = CALLS_APPLICATION (user_data); const gchar *number; gboolean number_ok; gchar *dial_string; number = g_variant_get_string (parameter, NULL); g_return_if_fail (number != NULL); number_ok = check_dial_number (number); if (!number_ok) { g_warning ("Dial number `%s' is not a valid dial string", number); return; } dial_string = extract_dial_string (number); if (!dial_string) { return; } g_debug ("Dialing dial string `%s' extracted from number `%s'", dial_string, number); start_proper (self); calls_main_window_dial (self->main_window, dial_string); g_free (dial_string); } static const GActionEntry actions[] = { { "set-provider-name", set_provider_name_action, "s" }, { "set-daemon", set_daemon_action, NULL }, { "dial", dial_action, "s" }, }; static void css_setup () { GtkCssProvider *provider; GFile *file; GError *error = NULL; provider = gtk_css_provider_new (); file = g_file_new_for_uri ("resource:///sm/puri/calls/style.css"); if (!gtk_css_provider_load_from_file (provider, file, &error)) { g_warning ("Failed to load CSS file: %s", error->message); g_clear_error (&error); g_object_unref (file); return; } gtk_style_context_add_provider_for_screen (gdk_screen_get_default (), GTK_STYLE_PROVIDER (provider), 600); g_object_unref (file); } static void startup (GApplication *application) { GtkIconTheme *icon_theme; g_autoptr(GError) error = NULL; G_APPLICATION_CLASS (calls_application_parent_class)->startup (application); hdy_init (); if (!call_audio_init (&error)) { g_warning ("Failed to init libcallaudio: %s", error->message); } g_set_prgname (APP_ID); g_set_application_name (_("Calls")); icon_theme = gtk_icon_theme_get_default (); gtk_icon_theme_add_resource_path (icon_theme, "/sm/puri/calls/"); g_action_map_add_action_entries (G_ACTION_MAP (application), actions, G_N_ELEMENTS (actions), application); css_setup (); } static void notify_window_visible_cb (GtkWidget *window, GParamSpec *pspec, CallsApplication *application) { CallsManager *manager = calls_manager_get_default (); g_return_if_fail (CALLS_IS_APPLICATION (application)); g_return_if_fail (CALLS_IS_CALL_WINDOW (window)); /* The UI is being closed, hang up active calls */ if (!gtk_widget_is_visible (window)) calls_manager_hang_up_all_calls (manager); } static gboolean start_proper (CallsApplication *self) { GtkApplication *gtk_app; if (self->main_window) { return TRUE; } gtk_app = GTK_APPLICATION (self); self->ringer = calls_ringer_new (); g_assert (self->ringer != NULL); self->record_store = calls_record_store_new (); g_assert (self->record_store != NULL); self->contacts = calls_contacts_get_default (); g_assert (self->contacts != NULL); self->notifier = calls_notifier_new (); g_assert (CALLS_IS_NOTIFIER (self->notifier)); self->main_window = calls_main_window_new (gtk_app, G_LIST_MODEL (self->record_store)); g_assert (self->main_window != NULL); self->call_window = calls_call_window_new (gtk_app); g_assert (self->call_window != NULL); g_signal_connect (self->call_window, "notify::visible", G_CALLBACK (notify_window_visible_cb), self); return TRUE; } static void activate (GApplication *application) { CallsApplication *self = CALLS_APPLICATION (application); gboolean present; g_debug ("Activated"); if (self->main_window) { present = TRUE; } else { gboolean ok = start_proper (self); if (!ok) { return; } present = !self->daemon; } if (present) { gtk_window_present (GTK_WINDOW (self->main_window)); } } static void open_tel_uri (CallsApplication *self, const gchar *uri) { EPhoneNumber *number; GError *error = NULL; gchar *dial_str; g_debug ("Opening tel URI `%s'", uri); number = e_phone_number_from_string (uri, NULL, &error); if (!number) { g_warning ("Ignoring unparsable tel URI `%s': %s", uri, error->message); g_error_free (error); return; } dial_str = e_phone_number_to_string (number, E_PHONE_NUMBER_FORMAT_E164); e_phone_number_free (number); calls_main_window_dial (self->main_window, dial_str); g_free (dial_str); } static void app_open (GApplication *application, GFile **files, gint n_files, const gchar *hint) { CallsApplication *self = CALLS_APPLICATION (application); gint i; g_assert (n_files > 0); g_debug ("Opened (%i files)", n_files); start_proper (self); for (i = 0; i < n_files; ++i) { gchar *uri; if (g_file_has_uri_scheme (files[i], "tel")) { uri = g_file_get_uri (files[i]); open_tel_uri (self, uri); } else { uri = g_file_get_parse_name (files[i]); g_warning ("Don't know how to" " open file `%s', ignoring", uri); } g_free (uri); } } static void constructed (GObject *object) { CallsApplication *self = CALLS_APPLICATION (object); GSimpleActionGroup *action_group; action_group = g_simple_action_group_new (); g_action_map_add_action_entries (G_ACTION_MAP (action_group), actions, G_N_ELEMENTS (actions), self); g_object_unref (action_group); G_OBJECT_CLASS (calls_application_parent_class)->constructed (object); } static void finalize (GObject *object) { CallsApplication *self = (CallsApplication *)object; g_clear_object (&self->call_window); g_clear_object (&self->main_window); g_clear_object (&self->record_store); g_clear_object (&self->contacts); g_clear_object (&self->ringer); g_clear_object (&self->notifier); G_OBJECT_CLASS (calls_application_parent_class)->dispose (object); } static void calls_application_class_init (CallsApplicationClass *klass) { GApplicationClass *application_class = G_APPLICATION_CLASS (klass); GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->constructed = constructed; object_class->finalize = finalize; application_class->handle_local_options = handle_local_options; application_class->startup = startup; application_class->activate = activate; application_class->open = app_open; g_type_ensure (CALLS_TYPE_ENCRYPTION_INDICATOR); g_type_ensure (CALLS_TYPE_HISTORY_BOX); g_type_ensure (CALLS_TYPE_NEW_CALL_BOX); } static void calls_application_init (CallsApplication *self) { const GOptionEntry options[] = { { "provider", 'p', G_OPTION_FLAG_NONE, G_OPTION_ARG_STRING, NULL, _("The name of the plugin to use for the call Provider"), _("PLUGIN") }, { "daemon", 'd', G_OPTION_FLAG_NONE, G_OPTION_ARG_NONE, NULL, _("Whether to present the main window on startup"), NULL }, { "dial", 'l', G_OPTION_FLAG_NONE, G_OPTION_ARG_STRING, NULL, _("Dial a number"), _("NUMBER") }, { NULL } }; g_application_add_main_option_entries (G_APPLICATION (self), options); } CallsApplication * calls_application_new (void) { return g_object_new (CALLS_TYPE_APPLICATION, "application-id", APP_ID, "flags", G_APPLICATION_HANDLES_OPEN, "register-session", TRUE, NULL); }