calls-application.c 12.1 KB
Newer Older
Mohammed Sadiq's avatar
Mohammed Sadiq committed
1
2
/* calls-application.c
 *
3
 * Copyright (C) 2018, 2019 Purism SPC
Mohammed Sadiq's avatar
Mohammed Sadiq committed
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
 * Copyright (C) 2018 Mohammed Sadiq <sadiq@sadiqpk.org>
 *
 * 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 <http://www.gnu.org/licenses/>.
 *
 * Authors:
 *      Bob Ham <bob.ham@puri.sm>
 *      Mohammed Sadiq <sadiq@sadiqpk.org>
 *
 * 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"
Bob Ham's avatar
Bob Ham committed
32
#include "calls-ringer.h"
33
#include "calls-record-store.h"
Mohammed Sadiq's avatar
Mohammed Sadiq committed
34
35
36
#include "calls-call-window.h"
#include "calls-main-window.h"
#include "calls-application.h"
37
#include "session.h"
Mohammed Sadiq's avatar
Mohammed Sadiq committed
38

39
40
41
42
43
#define HANDY_USE_UNSTABLE_API
#include <handy.h>

#include <libpeas/peas.h>
#include <glib/gi18n.h>
Bob Ham's avatar
Bob Ham committed
44
#include <libebook-contacts/libebook-contacts.h>
45

Mohammed Sadiq's avatar
Mohammed Sadiq committed
46
47
48
49
50
51
52
/**
 * SECTION: calls-application
 * @title: CallsApplication
 * @short_description: Base Application class
 * @include: "calls-application.h"
 */

53
54
#define DEFAULT_PROVIDER_PLUGIN "mm"

Mohammed Sadiq's avatar
Mohammed Sadiq committed
55
56
57
58
struct _CallsApplication
{
  GtkApplication parent_instance;

59
  gboolean          daemon;
60
61
62
63
  GString          *provider_name;
  CallsProvider    *provider;
  CallsRinger      *ringer;
  CallsRecordStore *record_store;
Bob Ham's avatar
Bob Ham committed
64
65
  CallsMainWindow  *main_window;
  CallsCallWindow  *call_window;
Mohammed Sadiq's avatar
Mohammed Sadiq committed
66
67
68
69
70
};

G_DEFINE_TYPE (CallsApplication, calls_application, GTK_TYPE_APPLICATION)


71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
static gint
handle_local_options (GApplication *application,
                      GVariantDict *options)
{
  gboolean ok;
  g_autoptr(GError) error = NULL;
  const gchar *name;

  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", &name);
  if (ok)
    {
      g_action_group_activate_action (G_ACTION_GROUP (application),
                                      "set-provider-name",
                                      g_variant_new_string (name));
    }

95
96
97
98
99
100
101
102
  ok = g_variant_dict_contains (options, "daemon");
  if (ok)
    {
      g_action_group_activate_action (G_ACTION_GROUP (application),
                                      "set-daemon",
                                      NULL);
    }

103
104
105
106
  return -1; // Continue processing signal
}


Mohammed Sadiq's avatar
Mohammed Sadiq committed
107
static void
108
109
110
set_provider_name_action (GSimpleAction *action,
                          GVariant      *parameter,
                          gpointer       user_data)
Mohammed Sadiq's avatar
Mohammed Sadiq committed
111
{
112
113
  CallsApplication *self = CALLS_APPLICATION (user_data);
  const gchar *name;
Mohammed Sadiq's avatar
Mohammed Sadiq committed
114

115
116
  name = g_variant_get_string (parameter, NULL);
  g_return_if_fail (name != NULL);
Mohammed Sadiq's avatar
Mohammed Sadiq committed
117

118
119
120
121
122
123
124
125
126
127
128
129
  if (self->provider)
    {
      g_warning ("Cannot set provider name to `%s'"
                 " because provider is already created",
                 name);
      return;
    }

  g_string_assign (self->provider_name, name);

  g_debug ("Provider name set to `%s'",
           self->provider_name->str);
Mohammed Sadiq's avatar
Mohammed Sadiq committed
130
131
}

132

133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
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");
}


153
154
155
static const GActionEntry actions[] =
{
  { "set-provider-name", set_provider_name_action, "s" },
156
  { "set-daemon", set_daemon_action, NULL },
157
158
159
};


Mohammed Sadiq's avatar
Mohammed Sadiq committed
160
161
162
static void
startup (GApplication *application)
{
163
164
  GtkIconTheme *icon_theme;

Mohammed Sadiq's avatar
Mohammed Sadiq committed
165
166
  G_APPLICATION_CLASS (calls_application_parent_class)->startup (application);

Bob Ham's avatar
Bob Ham committed
167
168
169
  g_set_prgname (APP_ID);
  g_set_application_name (_("Calls"));

170
171
172
  icon_theme = gtk_icon_theme_get_default ();
  gtk_icon_theme_add_resource_path (icon_theme, "/sm/puri/calls/");

173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
  g_action_map_add_action_entries (G_ACTION_MAP (application),
                                   actions,
                                   G_N_ELEMENTS (actions),
                                   application);
}


static void
load_provider_plugin (CallsApplication *self)
{
  const gchar * const name = self->provider_name->str;
  PeasEngine *plugins;
  PeasPluginInfo *info;
  PeasExtension *extension;

  g_assert (self->provider == NULL);

  // Add Calls search path and rescan
  plugins = peas_engine_get_default ();
  peas_engine_add_search_path (plugins, PLUGIN_LIBDIR, PLUGIN_LIBDIR);
  g_debug ("Scanning for plugins in `%s'", PLUGIN_LIBDIR);

  // Find the plugin
  info = peas_engine_get_plugin_info (plugins, name);
  if (!info)
    {
      g_critical ("Could not find plugin `%s'", name);
      return;
    }

  // Possibly load the plugin
  if (!peas_plugin_info_is_loaded (info))
    {
      g_autoptr(GError) error = NULL;
Mohammed Sadiq's avatar
Mohammed Sadiq committed
207

208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
      peas_engine_load_plugin (plugins, info);

      if (!peas_plugin_info_is_available (info, &error))
        {
          if (error)
            {
              g_critical ("Error loading plugin `%s': %s",
                          name, error->message);
            }
          else
            {
              g_critical ("Could not load plugin `%s'", name);
            }

          return;
        }

      g_debug ("Loaded plugin `%s'", name);
    }

  // Check the plugin provides CallsProvider
  if (!peas_engine_provides_extension
      (plugins, info, CALLS_TYPE_PROVIDER))
    {
      g_critical ("Plugin `%s' does not have a provider extension",
                  name);
      return;
    }

  // Get the extension
  extension = peas_engine_create_extensionv
    (plugins, info, CALLS_TYPE_PROVIDER, 0, NULL);
  if (!extension)
    {
      g_critical ("Could not create provider from plugin `%s'",
                  name);
      return;
    }

  g_debug ("Created provider from plugin `%s'", name);
  self->provider = CALLS_PROVIDER (extension);
Mohammed Sadiq's avatar
Mohammed Sadiq committed
249
250
}

251

Bob Ham's avatar
Bob Ham committed
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
static gboolean
start_proper (CallsApplication  *self)
{
  GtkApplication *gtk_app;

  if (self->main_window)
    {
      return TRUE;
    }

  gtk_app = GTK_APPLICATION (self);

  // Later we will make provider loading/unloaded a dynamic
  // process but that will have far-reaching consequences and is
  // of no use immediately so for now, we just load one provider
  // at startup.  We can't put this in the actual startup() method
  // though, because we need to be able to set the provider name
  // from the command line and we use actions to do that, which
  // depend on the application already being started up.
  load_provider_plugin (self);
  if (!self->provider)
    {
      g_application_quit (G_APPLICATION (self));
      return FALSE;
    }

  self->ringer = calls_ringer_new (self->provider);
  g_assert (self->ringer != NULL);

  self->record_store = calls_record_store_new (self->provider);
  g_assert (self->record_store != NULL);

  self->main_window = calls_main_window_new
    (gtk_app,
     self->provider,
     G_LIST_MODEL (self->record_store));
  g_assert (self->main_window != NULL);

  self->call_window = calls_call_window_new
    (gtk_app, self->provider);
  g_assert (self->call_window != NULL);

  return TRUE;
}


Mohammed Sadiq's avatar
Mohammed Sadiq committed
298
299
300
static void
activate (GApplication *application)
{
Bob Ham's avatar
Bob Ham committed
301
  CallsApplication *self = CALLS_APPLICATION (application);
302
  gboolean present;
Bob Ham's avatar
Bob Ham committed
303
304

  g_debug ("Activated");
Mohammed Sadiq's avatar
Mohammed Sadiq committed
305

306
  if (self->main_window)
Bob Ham's avatar
Bob Ham committed
307
    {
308
309
310
311
312
313
314
315
316
317
318
      present = TRUE;
    }
  else
    {
      gboolean ok = start_proper (self);
      if (!ok)
        {
          return;
        }

      present = !self->daemon;
Bob Ham's avatar
Bob Ham committed
319
320
    }

321
322
323
324
  if (present)
    {
      gtk_window_present (GTK_WINDOW (self->main_window));
    }
Bob Ham's avatar
Bob Ham committed
325
}
Mohammed Sadiq's avatar
Mohammed Sadiq committed
326
327


Bob Ham's avatar
Bob Ham committed
328
329
330
331
332
333
334
335
336
337
338
339
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)
Mohammed Sadiq's avatar
Mohammed Sadiq committed
340
    {
Bob Ham's avatar
Bob Ham committed
341
342
343
344
345
346
347
348
349
350
351
352
353
354
      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);
}
355

356

Bob Ham's avatar
Bob Ham committed
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
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);
387
388
        }

Bob Ham's avatar
Bob Ham committed
389
      g_free (uri);
Mohammed Sadiq's avatar
Mohammed Sadiq committed
390
391
392
    }
}

393
394
395
396
397
398
399
400
401
402
403
404
405

static void
constructed (GObject *object)
{
  GObjectClass *parent_class = g_type_class_peek (GTK_TYPE_APPLICATION);
  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);

406
407
  calls_session_register (APP_ID);

408
409
410
411
412
413
414
415
416
  parent_class->constructed (object);
}


static void
dispose (GObject *object)
{
  CallsApplication *self = (CallsApplication *)object;

417
418
  calls_session_unregister ();

Bob Ham's avatar
Bob Ham committed
419
420
  g_clear_object (&self->call_window);
  g_clear_object (&self->main_window);
421
  g_clear_object (&self->record_store);
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
  g_clear_object (&self->ringer);
  g_clear_object (&self->provider);

  G_OBJECT_CLASS (calls_application_parent_class)->dispose (object);
}


static void
finalize (GObject *object)
{
  CallsApplication *self = (CallsApplication *)object;

  g_string_free (self->provider_name, TRUE);

  G_OBJECT_CLASS (calls_application_parent_class)->finalize (object);
}


Mohammed Sadiq's avatar
Mohammed Sadiq committed
440
441
442
443
444
445
static void
calls_application_class_init (CallsApplicationClass *klass)
{
  GApplicationClass *application_class = G_APPLICATION_CLASS (klass);
  GObjectClass *object_class = G_OBJECT_CLASS (klass);

446
  object_class->constructed = constructed;
447
  object_class->dispose = dispose;
448
  object_class->finalize = finalize;
Mohammed Sadiq's avatar
Mohammed Sadiq committed
449

450
  application_class->handle_local_options = handle_local_options;
Mohammed Sadiq's avatar
Mohammed Sadiq committed
451
452
  application_class->startup = startup;
  application_class->activate = activate;
Bob Ham's avatar
Bob Ham committed
453
  application_class->open = app_open;
Mohammed Sadiq's avatar
Mohammed Sadiq committed
454
455
456
457
458

  g_type_ensure (CALLS_TYPE_ENCRYPTION_INDICATOR);
  g_type_ensure (CALLS_TYPE_HISTORY_BOX);
  g_type_ensure (CALLS_TYPE_NEW_CALL_BOX);
  g_type_ensure (HDY_TYPE_DIALER);
459
460
461
462
  g_type_ensure (HDY_TYPE_HEADER_BAR);
  g_type_ensure (HDY_TYPE_SQUEEZER);
  g_type_ensure (HDY_TYPE_VIEW_SWITCHER);
  g_type_ensure (HDY_TYPE_VIEW_SWITCHER_BAR);
Mohammed Sadiq's avatar
Mohammed Sadiq committed
463
464
}

465

Mohammed Sadiq's avatar
Mohammed Sadiq committed
466
467
468
static void
calls_application_init (CallsApplication *self)
{
469
470
471
472
473
474
475
  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")
    },
476
477
478
479
480
481
    {
      "daemon", 'd', G_OPTION_FLAG_NONE,
      G_OPTION_ARG_NONE, NULL,
      _("Whether to present the main window on startup"),
      NULL
    },
482
483
484
485
486
487
488
489
    {
      NULL
    }
  };

  g_application_add_main_option_entries (G_APPLICATION (self), options);

  self->provider_name = g_string_new (DEFAULT_PROVIDER_PLUGIN);
Mohammed Sadiq's avatar
Mohammed Sadiq committed
490
491
}

492

Mohammed Sadiq's avatar
Mohammed Sadiq committed
493
494
495
496
497
CallsApplication *
calls_application_new (void)
{
  return g_object_new (CALLS_TYPE_APPLICATION,
                       "application-id", APP_ID,
Bob Ham's avatar
Bob Ham committed
498
                       "flags", G_APPLICATION_HANDLES_OPEN,
Mohammed Sadiq's avatar
Mohammed Sadiq committed
499
500
                       NULL);
}