2020-03-17 15:29:43 +00:00
/*
* Copyright ( C ) 2020 Purism SPC
*
* 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/>.
*
* Author : Julian Sparber < julian . sparber @ puri . sm >
2021-07-03 22:20:57 +00:00
* Evangelos Ribeiro Tzaras < evangelos . tzaras @ puri . sm >
2020-03-17 15:29:43 +00:00
*
* SPDX - License - Identifier : GPL - 3.0 - or - later
*
*/
2021-06-18 09:10:36 +00:00
# define G_LOG_DOMAIN "CallsManager"
2022-12-18 10:18:10 +00:00
# include "calls-config.h"
2021-05-05 18:53:37 +00:00
2021-12-28 14:32:22 +00:00
# include "calls-account.h"
2021-01-20 14:20:09 +00:00
# include "calls-contacts-provider.h"
2021-05-05 18:53:37 +00:00
# include "calls-manager.h"
2021-12-28 13:43:54 +00:00
# include "calls-message-source.h"
2021-05-05 18:53:37 +00:00
# include "calls-provider.h"
2021-09-23 07:32:37 +00:00
# include "calls-settings.h"
2022-01-13 14:55:03 +00:00
# include "calls-ui-call-data.h"
2021-05-05 18:53:37 +00:00
# include "calls-ussd.h"
2023-01-21 15:42:39 +00:00
# include "calls-util.h"
2021-05-05 18:53:37 +00:00
2020-03-17 15:29:43 +00:00
# include "enum-types.h"
# include <glib/gi18n.h>
# include <libpeas/peas.h>
2022-02-06 08:56:31 +00:00
/**
* SECTION : manager
* @ short_description : Central management object
* @ Title : CallsManager
*
* # CallsManager is a singleton that manages lists of loaded # CallsProvider ,
* # CallsOrigin and # CallsCall objects . It keeps track of which # CallsOrigin
* supports which protocol . It also checks which # CallsCall are ongoing and
* emits signals for the UI and other parts of the application to act on .
*/
2022-01-26 06:53:58 +00:00
static const char * const protocols [ ] = {
2022-01-26 07:24:09 +00:00
" tel " ,
" sip " ,
" sips "
} ;
2022-04-24 10:24:55 +00:00
struct _CallsManager {
GObject parent_instance ;
2020-03-17 15:29:43 +00:00
2022-04-24 10:24:55 +00:00
GHashTable * providers ;
2021-04-13 09:36:58 +00:00
2022-04-24 10:24:55 +00:00
GListStore * origins ;
2021-04-13 09:36:58 +00:00
/* origins_by_protocol maps protocol names to GListStore's of suitable origins */
2022-04-24 10:24:55 +00:00
GHashTable * origins_by_protocol ;
2022-01-26 06:53:58 +00:00
/* dial_actions_by_protocol maps protocol names to GSimpleActions */
2022-04-24 10:24:55 +00:00
GHashTable * dial_actions_by_protocol ;
2021-04-13 09:36:58 +00:00
2022-01-13 14:55:03 +00:00
/* map CallsCall to CallsUiCallData */
2022-04-24 10:24:55 +00:00
GHashTable * calls ;
2022-01-13 14:55:03 +00:00
2021-01-20 14:20:09 +00:00
CallsContactsProvider * contacts_provider ;
2021-04-13 09:36:58 +00:00
2022-04-24 10:24:55 +00:00
CallsManagerFlags state_flags ;
CallsSettings * settings ;
2020-03-17 15:29:43 +00:00
} ;
2021-12-28 13:43:54 +00:00
static void
calls_manager_message_source_interface_init ( CallsMessageSourceInterface * iface )
{
}
G_DEFINE_TYPE_WITH_CODE ( CallsManager , calls_manager , G_TYPE_OBJECT ,
G_IMPLEMENT_INTERFACE ( CALLS_TYPE_MESSAGE_SOURCE ,
calls_manager_message_source_interface_init ) )
2020-03-17 15:29:43 +00:00
enum {
PROP_0 ,
2022-01-26 13:18:24 +00:00
PROP_STATE_FLAGS ,
2020-03-17 15:29:43 +00:00
PROP_LAST_PROP ,
} ;
static GParamSpec * props [ PROP_LAST_PROP ] ;
enum {
2022-02-04 03:46:00 +00:00
UI_CALL_ADDDED ,
2022-01-13 14:55:03 +00:00
UI_CALL_REMOVED ,
2020-07-07 12:40:31 +00:00
USSD_ADDED ,
USSD_CANCELLED ,
USSD_STATE_CHANGED ,
2021-07-03 23:42:04 +00:00
PROVIDERS_CHANGED ,
2020-03-17 15:29:43 +00:00
SIGNAL_LAST_SIGNAL ,
} ;
2022-04-24 10:24:55 +00:00
static guint signals [ SIGNAL_LAST_SIGNAL ] ;
2020-03-17 15:29:43 +00:00
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
2022-01-26 13:18:24 +00:00
set_state_flags ( CallsManager * self , CallsManagerFlags state_flags )
2020-03-17 15:29:43 +00:00
{
2022-01-26 13:18:24 +00:00
if ( self - > state_flags = = state_flags )
2020-03-17 15:29:43 +00:00
return ;
2022-01-26 13:18:24 +00:00
self - > state_flags = state_flags ;
2020-03-17 15:29:43 +00:00
2022-01-26 13:18:24 +00:00
g_object_notify_by_pspec ( G_OBJECT ( self ) , props [ PROP_STATE_FLAGS ] ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2021-04-13 09:36:58 +00:00
static void
2022-01-26 13:18:24 +00:00
update_state_flags ( CallsManager * self )
2021-04-13 09:36:58 +00:00
{
2021-05-04 19:49:07 +00:00
GHashTableIter iter ;
gpointer value ;
2022-01-26 13:18:24 +00:00
CallsManagerFlags state_flags = CALLS_MANAGER_FLAGS_UNKNOWN ;
2021-05-04 19:49:07 +00:00
2021-04-13 09:36:58 +00:00
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
2021-05-04 19:49:07 +00:00
g_hash_table_iter_init ( & iter , self - > providers ) ;
2022-01-26 07:32:35 +00:00
while ( g_hash_table_iter_next ( & iter , NULL , & value ) ) {
2021-05-04 19:49:07 +00:00
CallsProvider * provider = CALLS_PROVIDER ( value ) ;
2022-01-26 13:18:24 +00:00
if ( calls_provider_is_modem ( provider ) ) {
state_flags | = CALLS_MANAGER_FLAGS_HAS_CELLULAR_PROVIDER ;
2021-04-13 09:36:58 +00:00
2022-01-26 13:18:24 +00:00
if ( calls_provider_is_operational ( provider ) )
state_flags | = CALLS_MANAGER_FLAGS_HAS_CELLULAR_MODEM ;
} else {
state_flags | = CALLS_MANAGER_FLAGS_HAS_VOIP_PROVIDER ;
2021-04-13 09:36:58 +00:00
2022-01-26 13:18:24 +00:00
if ( calls_provider_is_operational ( provider ) )
state_flags | = CALLS_MANAGER_FLAGS_HAS_VOIP_ACCOUNT ;
}
}
set_state_flags ( self , state_flags ) ;
2021-04-13 09:36:58 +00:00
}
2021-12-27 14:17:19 +00:00
2022-01-26 06:53:58 +00:00
static void
on_dial_protocol_activated ( GSimpleAction * action ,
GVariant * parameter ,
CallsManager * self )
{
GApplication * application ;
CallsOrigin * origin ;
g_autofree char * target = NULL ;
g_autofree char * origin_id = NULL ;
g_variant_get ( parameter , " (ss) " , & target , & origin_id ) ;
2022-02-10 09:12:16 +00:00
origin = calls_manager_get_origin_by_id ( self , origin_id ) ;
2022-01-26 06:53:58 +00:00
if ( origin ) {
calls_origin_dial ( origin , target ) ;
return ;
}
if ( origin_id & & * origin_id )
g_debug ( " Origin ID '%s' given, but was not found for call to '%s' " ,
origin_id , target ) ;
/* fall back to the default action if we could not determine origin to place call from */
application = g_application_get_default ( ) ;
if ( ! application ) {
g_warning ( " Could not get default application, cannot activate action '%s' " ,
g_action_get_name ( G_ACTION ( action ) ) ) ;
return ;
}
g_action_group_activate_action ( G_ACTION_GROUP ( application ) ,
" dial " ,
g_variant_new_string ( target ) ) ;
}
static void
update_protocol_dial_actions ( CallsManager * self )
{
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
for ( guint i = 0 ; i < G_N_ELEMENTS ( protocols ) ; i + + ) {
GSimpleAction * action = g_hash_table_lookup ( self - > dial_actions_by_protocol ,
protocols [ i ] ) ;
GListModel * protocol_origin = g_hash_table_lookup ( self - > origins_by_protocol ,
protocols [ i ] ) ;
/* TODO take into account if origin is active: modem registered or VoIP account online */
gboolean action_enabled = g_list_model_get_n_items ( protocol_origin ) > 0 ;
g_simple_action_set_enabled ( action , action_enabled ) ;
}
}
2021-12-28 14:32:22 +00:00
static void
on_message ( CallsMessageSource * source ,
const char * message ,
GtkMessageType message_type ,
CallsManager * self )
{
g_autofree char * notification = NULL ;
g_assert ( CALLS_IS_MESSAGE_SOURCE ( source ) ) ;
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
/* Prefix the message with the name of the source, if known */
if ( CALLS_IS_ACCOUNT ( source ) ) {
notification = g_strdup_printf ( " %s: %s " ,
calls_account_get_address ( CALLS_ACCOUNT ( source ) ) ,
message ) ;
}
2022-01-13 15:03:38 +00:00
calls_message_source_emit_message ( CALLS_MESSAGE_SOURCE ( self ) ,
2022-04-24 10:24:55 +00:00
notification ? : message ,
2022-01-13 15:03:38 +00:00
message_type ) ;
2021-12-28 14:32:22 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
add_call ( CallsManager * self , CallsCall * call , CallsOrigin * origin )
{
2022-01-13 14:55:03 +00:00
CallsUiCallData * call_data ;
2022-02-03 17:52:52 +00:00
g_autofree char * origin_id = NULL ;
2022-01-13 14:55:03 +00:00
2020-03-17 15:29:43 +00:00
g_return_if_fail ( CALLS_IS_MANAGER ( self ) ) ;
g_return_if_fail ( CALLS_IS_ORIGIN ( origin ) ) ;
g_return_if_fail ( CALLS_IS_CALL ( call ) ) ;
2022-02-03 17:52:52 +00:00
origin_id = calls_origin_get_id ( origin ) ;
call_data = calls_ui_call_data_new ( call , origin_id ) ;
2022-01-13 14:55:03 +00:00
g_hash_table_insert ( self - > calls , call , call_data ) ;
g_signal_emit ( self , signals [ UI_CALL_ADDDED ] , 0 , call_data ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2022-04-24 10:24:55 +00:00
struct CallsRemoveData {
2022-02-03 06:49:55 +00:00
CallsManager * manager ;
CallsCall * call ;
} ;
static gboolean
on_remove_delayed ( struct CallsRemoveData * data )
{
CallsUiCallData * call_data ;
g_assert ( CALLS_IS_MANAGER ( data - > manager ) ) ;
g_assert ( CALLS_IS_CALL ( data - > call ) ) ;
call_data = g_hash_table_lookup ( data - > manager - > calls , data - > call ) ;
if ( ! call_data ) {
g_warning ( " Could not find call %s in UI call hash table " ,
calls_call_get_id ( data - > call ) ) ;
} else {
g_signal_emit ( data - > manager , signals [ UI_CALL_REMOVED ] , 0 , call_data ) ;
g_hash_table_remove ( data - > manager - > calls , data - > call ) ;
}
g_free ( data ) ;
return G_SOURCE_REMOVE ;
}
# define DELAY_CALL_REMOVE_MS 3000
2020-03-17 15:29:43 +00:00
static void
remove_call ( CallsManager * self , CallsCall * call , gchar * reason , CallsOrigin * origin )
{
2022-02-03 06:49:55 +00:00
struct CallsRemoveData * data ;
2022-01-13 14:55:03 +00:00
2020-03-17 15:29:43 +00:00
g_return_if_fail ( CALLS_IS_MANAGER ( self ) ) ;
g_return_if_fail ( CALLS_IS_ORIGIN ( origin ) ) ;
g_return_if_fail ( CALLS_IS_CALL ( call ) ) ;
2022-02-03 06:49:55 +00:00
data = g_new ( struct CallsRemoveData , 1 ) ;
data - > manager = self ;
data - > call = call ;
/** Having the delay centrally managed makes sure our UI
* and the DBus side stays consistent
*/
g_timeout_add ( DELAY_CALL_REMOVE_MS ,
G_SOURCE_FUNC ( on_remove_delayed ) ,
data ) ;
2020-03-17 15:29:43 +00:00
}
2022-02-03 06:49:55 +00:00
# undef DELAY_CALL_REMOVE_MS
2020-03-17 15:29:43 +00:00
2021-12-27 14:17:19 +00:00
2020-07-07 12:40:31 +00:00
static void
ussd_added_cb ( CallsManager * self ,
char * response ,
CallsUssd * ussd )
{
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( CALLS_IS_USSD ( ussd ) ) ;
g_signal_emit ( self , signals [ USSD_ADDED ] , 0 , ussd , response ) ;
}
2021-12-27 14:17:19 +00:00
2020-07-07 12:40:31 +00:00
static void
ussd_cancelled_cb ( CallsManager * self ,
CallsUssd * ussd ,
char * response )
{
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( CALLS_IS_USSD ( ussd ) ) ;
g_signal_emit ( self , signals [ USSD_CANCELLED ] , 0 , ussd ) ;
}
2021-12-27 14:17:19 +00:00
2020-07-07 12:40:31 +00:00
static void
ussd_state_changed_cb ( CallsManager * self ,
CallsUssd * ussd )
{
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( CALLS_IS_USSD ( ussd ) ) ;
g_signal_emit ( self , signals [ USSD_STATE_CHANGED ] , 0 , ussd ) ;
}
2021-12-27 14:17:19 +00:00
2021-05-18 16:31:05 +00:00
static void
update_country_code_cb ( CallsOrigin * origin ,
GParamSpec * pspec ,
CallsManager * self )
{
g_autofree char * country_code = NULL ;
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_object_get ( G_OBJECT ( origin ) , " country-code " , & country_code , NULL ) ;
2021-09-23 07:32:37 +00:00
if ( ! country_code | | ! * country_code )
2021-05-18 16:31:05 +00:00
return ;
2021-09-23 07:32:37 +00:00
calls_settings_set_country_code ( self - > settings , country_code ) ;
2021-05-18 16:31:05 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
2021-04-18 00:33:13 +00:00
add_origin ( CallsManager * self , CallsOrigin * origin )
2020-03-17 15:29:43 +00:00
{
2021-05-13 15:35:10 +00:00
g_autofree const char * name = NULL ;
2022-04-24 10:24:55 +00:00
2021-04-14 14:05:40 +00:00
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( CALLS_IS_ORIGIN ( origin ) ) ;
2020-03-17 15:29:43 +00:00
2021-05-13 15:35:10 +00:00
name = calls_origin_get_name ( origin ) ;
g_debug ( " Adding origin %s (%p) " , name , origin ) ;
2021-04-17 14:43:21 +00:00
2021-04-13 09:36:58 +00:00
g_list_store_append ( self - > origins , origin ) ;
2021-12-28 14:32:22 +00:00
g_signal_connect ( origin ,
" message " ,
G_CALLBACK ( on_message ) ,
self ) ;
2021-05-18 16:31:05 +00:00
g_signal_connect_object ( origin ,
" notify::country-code " ,
G_CALLBACK ( update_country_code_cb ) ,
self ,
G_CONNECT_AFTER ) ;
2020-03-17 15:29:43 +00:00
g_signal_connect_swapped ( origin , " call-added " , G_CALLBACK ( add_call ) , self ) ;
g_signal_connect_swapped ( origin , " call-removed " , G_CALLBACK ( remove_call ) , self ) ;
2022-04-24 10:24:55 +00:00
if ( CALLS_IS_USSD ( origin ) ) {
g_signal_connect_swapped ( origin , " ussd-added " , G_CALLBACK ( ussd_added_cb ) , self ) ;
g_signal_connect_swapped ( origin , " ussd-cancelled " , G_CALLBACK ( ussd_cancelled_cb ) , self ) ;
g_signal_connect_swapped ( origin , " ussd-state-changed " , G_CALLBACK ( ussd_state_changed_cb ) , self ) ;
}
2020-07-07 12:40:31 +00:00
2021-04-13 09:36:58 +00:00
calls_origin_foreach_call ( origin , ( CallsOriginForeachCallFunc ) add_call , self ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-23 22:10:05 +00:00
static void
remove_call_cb ( gpointer self , CallsCall * call , CallsOrigin * origin )
{
2021-04-13 09:36:58 +00:00
remove_call ( self , call , NULL , origin ) ;
2020-03-23 22:10:05 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
2021-04-18 00:33:13 +00:00
remove_origin ( CallsManager * self , CallsOrigin * origin )
2020-03-17 15:29:43 +00:00
{
2021-05-13 15:35:10 +00:00
g_autofree const char * name = NULL ;
2021-04-13 09:36:58 +00:00
guint position ;
2020-03-23 22:10:05 +00:00
2021-04-13 09:36:58 +00:00
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( CALLS_IS_ORIGIN ( origin ) ) ;
2020-03-17 15:29:43 +00:00
2021-05-13 15:35:10 +00:00
name = calls_origin_get_name ( origin ) ;
g_debug ( " Removing origin %s (%p) " , name , origin ) ;
2021-04-17 14:43:21 +00:00
2020-03-17 15:29:43 +00:00
g_signal_handlers_disconnect_by_data ( origin , self ) ;
2021-04-13 09:36:58 +00:00
calls_origin_foreach_call ( origin , remove_call_cb , self ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
if ( ! g_list_store_find ( self - > origins , origin , & position ) )
g_warning ( " Origin %p not found in list store while trying to remove it " ,
origin ) ;
else
g_list_store_remove ( self - > origins , position ) ;
}
/* rebuild_origins_by_protocols() when any origins were added or removed */
static void
rebuild_origins_by_protocols ( CallsManager * self )
{
GHashTableIter iter ;
2022-01-26 07:32:35 +00:00
gpointer value ;
2021-04-13 09:36:58 +00:00
guint n_origins ;
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
/* Remove everything */
g_hash_table_iter_init ( & iter , self - > origins_by_protocol ) ;
2022-01-26 07:32:35 +00:00
while ( g_hash_table_iter_next ( & iter , NULL , & value ) ) {
2021-04-13 09:36:58 +00:00
GListStore * store = G_LIST_STORE ( value ) ;
g_list_store_remove_all ( store ) ;
}
/* Iterate over all origins and check which protocols they support */
n_origins = g_list_model_get_n_items ( G_LIST_MODEL ( self - > origins ) ) ;
for ( guint i = 0 ; i < n_origins ; i + + ) {
g_autoptr ( CallsOrigin ) origin =
g_list_model_get_item ( G_LIST_MODEL ( self - > origins ) , i ) ;
2022-01-26 07:24:09 +00:00
for ( guint j = 0 ; j < G_N_ELEMENTS ( protocols ) ; j + + ) {
2021-04-13 09:36:58 +00:00
GListStore * store =
2022-01-26 07:24:09 +00:00
G_LIST_STORE ( g_hash_table_lookup ( self - > origins_by_protocol , protocols [ j ] ) ) ;
2021-04-13 09:36:58 +00:00
g_assert ( store ) ;
2022-01-26 07:24:09 +00:00
if ( calls_origin_supports_protocol ( origin , protocols [ j ] ) )
2021-04-13 09:36:58 +00:00
g_list_store_append ( store , origin ) ;
}
}
2022-01-26 06:53:58 +00:00
update_protocol_dial_actions ( self ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
2021-04-13 09:36:58 +00:00
remove_provider ( CallsManager * self ,
const char * name )
2020-03-17 15:29:43 +00:00
{
2022-04-24 10:24:55 +00:00
g_autoptr ( CallsProvider ) provider = NULL ;
2021-02-15 05:34:34 +00:00
GListModel * origins ;
guint n_items ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( name ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
provider = g_hash_table_lookup ( self - > providers , name ) ;
2021-07-07 14:12:23 +00:00
if ( provider ) {
/* Hold a ref since g_hash_table_remove () might drop the last one */
g_object_ref ( provider ) ;
} else {
2021-04-13 09:36:58 +00:00
g_warning ( " Trying to remove provider %s which has not been found " , name ) ;
return ;
}
g_debug ( " Remove provider: %s " , name ) ;
g_signal_handlers_disconnect_by_data ( provider , self ) ;
origins = calls_provider_get_origins ( provider ) ;
2021-02-15 05:34:34 +00:00
g_signal_handlers_disconnect_by_data ( origins , self ) ;
n_items = g_list_model_get_n_items ( origins ) ;
2020-03-17 15:29:43 +00:00
2021-06-09 02:03:31 +00:00
for ( guint i = 0 ; i < n_items ; i + + ) {
2022-04-24 10:24:55 +00:00
g_autoptr ( CallsOrigin ) origin = NULL ;
2021-02-15 05:34:34 +00:00
2021-06-09 02:03:31 +00:00
origin = g_list_model_get_item ( origins , i ) ;
remove_origin ( self , origin ) ;
}
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
g_hash_table_remove ( self - > providers , name ) ;
calls_provider_unload_plugin ( name ) ;
rebuild_origins_by_protocols ( self ) ;
2022-01-26 13:18:24 +00:00
update_state_flags ( self ) ;
2021-07-03 23:42:04 +00:00
g_signal_emit ( self , signals [ PROVIDERS_CHANGED ] , 0 ) ;
2021-04-13 09:36:58 +00:00
}
2021-12-27 14:17:19 +00:00
2021-04-13 09:36:58 +00:00
static gboolean
origin_found_in_any_provider ( CallsManager * self ,
CallsOrigin * origin )
{
GHashTableIter iter ;
2022-01-26 07:32:35 +00:00
gpointer value ;
2021-04-13 09:36:58 +00:00
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , FALSE ) ;
g_return_val_if_fail ( CALLS_IS_ORIGIN ( origin ) , FALSE ) ;
g_hash_table_iter_init ( & iter , self - > providers ) ;
2022-01-26 07:32:35 +00:00
while ( g_hash_table_iter_next ( & iter , NULL , & value ) ) {
2021-04-13 09:36:58 +00:00
guint position ;
CallsProvider * provider = CALLS_PROVIDER ( value ) ;
GListModel * origins = calls_provider_get_origins ( provider ) ;
2022-10-10 06:39:42 +00:00
if ( origins & & calls_find_in_model ( origins ,
2021-04-13 09:36:58 +00:00
origin ,
& position ) )
return TRUE ;
}
2021-04-02 14:04:07 +00:00
2021-04-13 09:36:58 +00:00
return FALSE ;
2020-03-17 15:29:43 +00:00
}
2021-04-13 09:36:58 +00:00
2021-02-15 05:34:34 +00:00
static void
2021-04-13 09:36:58 +00:00
origin_items_changed_cb ( GListModel * model ,
guint position ,
guint removed ,
guint added ,
CallsManager * self )
2021-02-15 05:34:34 +00:00
{
2021-04-13 09:36:58 +00:00
guint i ;
CallsOrigin * origin ;
guint purged = 0 ;
guint total_origins ;
2021-02-15 05:34:34 +00:00
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
2021-04-13 09:36:58 +00:00
total_origins = g_list_model_get_n_items ( G_LIST_MODEL ( self - > origins ) ) ;
g_debug ( " origins changed: pos=%d rem=%d added=%d total=%d " ,
position , removed , added , g_list_model_get_n_items ( model ) ) ;
2021-02-15 05:34:34 +00:00
2021-04-13 09:36:58 +00:00
/* Check stale/removed origins: We need to look up */
if ( removed = = 0 )
goto skip_remove ;
2021-02-15 05:34:34 +00:00
2021-04-13 09:36:58 +00:00
for ( i = 0 ; i < total_origins - purged ; i + + ) {
origin = g_list_model_get_item ( G_LIST_MODEL ( self - > origins ) , i - purged ) ;
2021-02-15 05:34:34 +00:00
2021-04-13 09:36:58 +00:00
if ( ! origin_found_in_any_provider ( self , origin ) ) {
remove_origin ( self , origin ) ;
purged + + ;
2021-02-15 05:34:34 +00:00
}
2021-04-13 09:36:58 +00:00
}
/** The number of purged entries from self->origins must be equal to removed
* origins from the providers list
*/
if ( purged ! = removed ) {
g_warning ( " Managed origins are not in sync anymore! " ) ;
}
2022-04-24 10:24:55 +00:00
skip_remove :
2021-04-13 09:36:58 +00:00
for ( i = 0 ; i < added ; i + + ) {
g_debug ( " before adding: %d " ,
g_list_model_get_n_items ( G_LIST_MODEL ( self - > origins ) ) ) ;
origin = g_list_model_get_item ( model , position + i ) ;
add_origin ( self , origin ) ; // add to list store
g_object_unref ( origin ) ;
g_debug ( " after adding: %d " ,
g_list_model_get_n_items ( G_LIST_MODEL ( self - > origins ) ) ) ;
}
rebuild_origins_by_protocols ( self ) ;
2022-01-26 13:18:24 +00:00
update_state_flags ( self ) ;
2021-02-15 05:34:34 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
add_provider ( CallsManager * self , const gchar * name )
{
2021-02-15 05:34:34 +00:00
GListModel * origins ;
2021-04-13 09:36:58 +00:00
CallsProvider * provider ;
guint n_items ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
g_assert ( CALLS_IS_MANAGER ( self ) ) ;
g_assert ( name ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
if ( g_hash_table_lookup ( self - > providers , name ) )
2020-03-17 15:29:43 +00:00
return ;
2021-04-13 09:36:58 +00:00
provider = calls_provider_load_plugin ( name ) ;
if ( provider = = NULL ) {
g_warning ( " Could not load a plugin with name `%s' " , name ) ;
2020-03-17 15:29:43 +00:00
return ;
}
2021-04-13 09:36:58 +00:00
g_hash_table_insert ( self - > providers , g_strdup ( name ) , provider ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
origins = calls_provider_get_origins ( provider ) ;
2020-03-17 15:29:43 +00:00
2021-02-15 05:34:34 +00:00
g_signal_connect_object ( origins , " items-changed " ,
G_CALLBACK ( origin_items_changed_cb ) , self ,
2021-04-13 09:36:58 +00:00
G_CONNECT_AFTER ) ;
n_items = g_list_model_get_n_items ( origins ) ;
origin_items_changed_cb ( origins , 0 , 0 , n_items , self ) ;
2020-03-17 15:29:43 +00:00
2021-07-03 23:42:04 +00:00
g_signal_emit ( self , signals [ PROVIDERS_CHANGED ] , 0 ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
static void
calls_manager_get_property ( GObject * object ,
guint property_id ,
GValue * value ,
GParamSpec * pspec )
{
CallsManager * self = CALLS_MANAGER ( object ) ;
switch ( property_id ) {
2022-01-26 13:18:24 +00:00
case PROP_STATE_FLAGS :
g_value_set_flags ( value , calls_manager_get_state_flags ( self ) ) ;
2020-03-17 15:29:43 +00:00
break ;
2021-01-11 23:00:20 +00:00
2020-03-17 15:29:43 +00:00
default :
G_OBJECT_WARN_INVALID_PROPERTY_ID ( object , property_id , pspec ) ;
break ;
}
2021-01-11 23:00:20 +00:00
}
2020-03-17 15:29:43 +00:00
static void
calls_manager_finalize ( GObject * object )
{
CallsManager * self = CALLS_MANAGER ( object ) ;
2021-04-13 09:36:58 +00:00
g_clear_object ( & self - > origins ) ;
2021-01-20 14:20:09 +00:00
g_clear_object ( & self - > contacts_provider ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
g_clear_pointer ( & self - > providers , g_hash_table_unref ) ;
g_clear_pointer ( & self - > origins_by_protocol , g_hash_table_unref ) ;
2022-01-26 06:53:58 +00:00
g_clear_pointer ( & self - > dial_actions_by_protocol , g_hash_table_unref ) ;
2021-04-13 09:36:58 +00:00
2020-03-17 15:29:43 +00:00
G_OBJECT_CLASS ( calls_manager_parent_class ) - > finalize ( object ) ;
}
static void
calls_manager_class_init ( CallsManagerClass * klass )
{
GObjectClass * object_class = G_OBJECT_CLASS ( klass ) ;
object_class - > get_property = calls_manager_get_property ;
object_class - > finalize = calls_manager_finalize ;
2022-01-13 14:55:03 +00:00
signals [ UI_CALL_ADDDED ] =
g_signal_new ( " ui-call-added " ,
G_TYPE_FROM_CLASS ( klass ) ,
G_SIGNAL_RUN_FIRST ,
0 ,
NULL , NULL , NULL ,
G_TYPE_NONE ,
1 ,
CALLS_TYPE_UI_CALL_DATA ) ;
signals [ UI_CALL_REMOVED ] =
g_signal_new ( " ui-call-removed " ,
G_TYPE_FROM_CLASS ( klass ) ,
G_SIGNAL_RUN_FIRST ,
0 ,
NULL , NULL , NULL ,
G_TYPE_NONE ,
1 ,
CALLS_TYPE_UI_CALL_DATA ) ;
2020-07-07 12:40:31 +00:00
signals [ USSD_ADDED ] =
g_signal_new ( " ussd-added " ,
G_TYPE_FROM_CLASS ( klass ) ,
G_SIGNAL_RUN_FIRST ,
0 ,
NULL , NULL , NULL ,
G_TYPE_NONE ,
2 ,
CALLS_TYPE_USSD ,
G_TYPE_STRING ) ;
signals [ USSD_CANCELLED ] =
g_signal_new ( " ussd-cancelled " ,
G_TYPE_FROM_CLASS ( klass ) ,
G_SIGNAL_RUN_FIRST ,
0 ,
NULL , NULL , NULL ,
G_TYPE_NONE ,
1 ,
CALLS_TYPE_USSD ) ;
signals [ USSD_STATE_CHANGED ] =
g_signal_new ( " ussd-state-changed " ,
G_TYPE_FROM_CLASS ( klass ) ,
G_SIGNAL_RUN_FIRST ,
0 ,
NULL , NULL , NULL ,
G_TYPE_NONE ,
1 ,
CALLS_TYPE_USSD ) ;
2021-07-03 23:42:04 +00:00
signals [ PROVIDERS_CHANGED ] =
g_signal_new ( " providers-changed " ,
G_TYPE_FROM_CLASS ( klass ) ,
G_SIGNAL_RUN_FIRST ,
0 ,
NULL , NULL , NULL ,
G_TYPE_NONE ,
0 ) ;
2022-01-26 13:18:24 +00:00
props [ PROP_STATE_FLAGS ] =
g_param_spec_flags ( " state " ,
" state " ,
" The state of the Manager " ,
CALLS_TYPE_MANAGER_FLAGS ,
CALLS_MANAGER_FLAGS_UNKNOWN ,
G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY ) ;
2020-03-17 15:29:43 +00:00
g_object_class_install_properties ( object_class , PROP_LAST_PROP , props ) ;
}
static void
calls_manager_init ( CallsManager * self )
{
2022-01-26 06:53:58 +00:00
g_autoptr ( GVariantType ) variant_type = NULL ;
GApplication * application ;
2021-04-21 09:31:29 +00:00
PeasEngine * peas ;
const gchar * dir ;
2022-07-18 20:30:19 +00:00
g_autofree char * default_plugin_dir_provider = NULL ;
2021-04-21 09:31:29 +00:00
2022-01-26 13:18:24 +00:00
self - > state_flags = CALLS_MANAGER_FLAGS_UNKNOWN ;
2021-04-13 09:36:58 +00:00
self - > providers = g_hash_table_new_full ( g_str_hash ,
g_str_equal ,
g_free ,
g_object_unref ) ;
self - > origins_by_protocol = g_hash_table_new_full ( g_str_hash ,
g_str_equal ,
g_free ,
g_object_unref ) ;
2022-01-26 07:24:09 +00:00
for ( guint i = 0 ; i < G_N_ELEMENTS ( protocols ) ; i + + ) {
GListStore * origin_store = g_list_store_new ( calls_origin_get_type ( ) ) ;
g_hash_table_insert ( self - > origins_by_protocol ,
g_strdup ( protocols [ i ] ) ,
origin_store ) ;
}
2022-01-26 06:53:58 +00:00
self - > dial_actions_by_protocol = g_hash_table_new_full ( g_str_hash ,
g_str_equal ,
g_free ,
g_object_unref ) ;
application = g_application_get_default ( ) ;
variant_type = g_variant_type_new ( " (ss) " ) ;
for ( guint i = 0 ; i < G_N_ELEMENTS ( protocols ) ; i + + ) {
g_autofree char * action_name = g_strdup_printf ( " dial-%s " , protocols [ i ] ) ;
GSimpleAction * action = g_simple_action_new ( action_name , variant_type ) ;
g_signal_connect ( action ,
" activate " ,
G_CALLBACK ( on_dial_protocol_activated ) ,
self ) ;
g_hash_table_insert ( self - > dial_actions_by_protocol ,
g_strdup ( protocols [ i ] ) ,
g_object_ref ( action ) ) ;
/* Enable action if there are suitable origins */
g_simple_action_set_enabled ( action , FALSE ) ;
/* application can be NULL when running tests */
if ( application )
g_action_map_add_action ( G_ACTION_MAP ( application ) , G_ACTION ( action ) ) ;
}
2021-04-13 09:36:58 +00:00
self - > origins = g_list_store_new ( calls_origin_get_type ( ) ) ;
2021-01-20 14:20:09 +00:00
2022-01-13 14:55:03 +00:00
/* This hash table only owns the value, not the key */
self - > calls = g_hash_table_new_full ( NULL , NULL , NULL , g_object_unref ) ;
2022-05-09 20:57:41 +00:00
self - > settings = calls_settings_get_default ( ) ;
2021-01-20 14:20:09 +00:00
// Load the contacts provider
2022-05-09 20:57:41 +00:00
self - > contacts_provider = calls_contacts_provider_new ( ) ;
2021-04-21 09:31:29 +00:00
peas = peas_engine_get_default ( ) ;
dir = g_getenv ( " CALLS_PLUGIN_DIR " ) ;
if ( dir & & dir [ 0 ] ! = ' \0 ' ) {
2022-07-18 20:30:19 +00:00
g_autofree char * plugin_dir_provider = NULL ;
plugin_dir_provider = g_build_filename ( dir , " provider " , NULL ) ;
2022-09-22 15:54:44 +00:00
if ( g_file_test ( plugin_dir_provider , G_FILE_TEST_EXISTS ) ) {
g_debug ( " Adding %s to plugin search path " , plugin_dir_provider ) ;
peas_engine_prepend_search_path ( peas , plugin_dir_provider , NULL ) ;
} else {
g_warning ( " Not adding %s to plugin search path, because the directory doesn't exist. Check if env CALLS_PLUGIN_DIR is set correctly " , plugin_dir_provider ) ;
}
2021-04-21 09:31:29 +00:00
}
2022-07-18 20:30:19 +00:00
default_plugin_dir_provider = g_build_filename ( PLUGIN_LIBDIR , " provider " , NULL ) ;
peas_engine_add_search_path ( peas , default_plugin_dir_provider , NULL ) ;
g_debug ( " Scanning for plugins in `%s' " , default_plugin_dir_provider ) ;
2022-10-26 13:09:11 +00:00
peas_engine_rescan_plugins ( peas ) ;
2020-03-17 15:29:43 +00:00
}
CallsManager *
calls_manager_new ( void )
{
return g_object_new ( CALLS_TYPE_MANAGER , NULL ) ;
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
CallsManager *
calls_manager_get_default ( void )
{
static CallsManager * instance ;
if ( instance = = NULL ) {
instance = calls_manager_new ( ) ;
2022-04-24 10:24:55 +00:00
g_object_add_weak_pointer ( G_OBJECT ( instance ) , ( gpointer * ) & instance ) ;
2020-03-17 15:29:43 +00:00
}
return instance ;
}
2021-12-27 14:17:19 +00:00
2021-01-20 14:20:09 +00:00
CallsContactsProvider *
calls_manager_get_contacts_provider ( CallsManager * self )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
return self - > contacts_provider ;
}
2021-12-27 14:17:19 +00:00
2021-04-13 09:36:58 +00:00
void
calls_manager_add_provider ( CallsManager * self ,
const char * name )
2020-03-17 15:29:43 +00:00
{
2021-04-13 09:36:58 +00:00
g_return_if_fail ( CALLS_IS_MANAGER ( self ) ) ;
g_return_if_fail ( name ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
add_provider ( self , name ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2020-03-17 15:29:43 +00:00
void
2021-04-13 09:36:58 +00:00
calls_manager_remove_provider ( CallsManager * self ,
const char * name )
2020-03-17 15:29:43 +00:00
{
g_return_if_fail ( CALLS_IS_MANAGER ( self ) ) ;
2021-04-13 09:36:58 +00:00
g_return_if_fail ( name ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
remove_provider ( self , name ) ;
}
2020-03-17 15:29:43 +00:00
2021-12-27 14:17:19 +00:00
2021-04-13 09:36:58 +00:00
gboolean
calls_manager_has_provider ( CallsManager * self ,
const char * name )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , FALSE ) ;
g_return_val_if_fail ( name , FALSE ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
return ! ! g_hash_table_lookup ( self - > providers , name ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2021-05-05 18:53:37 +00:00
gboolean
calls_manager_is_modem_provider ( CallsManager * self ,
const char * name )
{
CallsProvider * provider ;
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , FALSE ) ;
g_return_val_if_fail ( name , FALSE ) ;
provider = g_hash_table_lookup ( self - > providers , name ) ;
g_return_val_if_fail ( provider , FALSE ) ;
return calls_provider_is_modem ( provider ) ;
}
2022-01-26 13:18:24 +00:00
CallsManagerFlags
calls_manager_get_state_flags ( CallsManager * self )
2020-03-17 15:29:43 +00:00
{
2022-01-26 13:18:24 +00:00
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , CALLS_MANAGER_FLAGS_UNKNOWN ) ;
2020-03-17 15:29:43 +00:00
2022-01-26 13:18:24 +00:00
return self - > state_flags ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2021-02-15 05:34:34 +00:00
GListModel *
2020-03-17 15:29:43 +00:00
calls_manager_get_origins ( CallsManager * self )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
2021-04-13 09:36:58 +00:00
return G_LIST_MODEL ( self - > origins ) ;
2020-03-17 15:29:43 +00:00
}
2021-12-27 14:17:19 +00:00
2022-02-01 05:08:12 +00:00
/**
* calls_manager_get_calls :
* @ self : A # CallsManager
*
* Returns : ( transfer container ) : Returns a list of all known calls .
* The calls are objects of type # CallsUiCallData . Use g_list_free ( ) when
* done using the list .
*/
2020-03-17 15:29:43 +00:00
GList *
calls_manager_get_calls ( CallsManager * self )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
2022-02-01 05:08:12 +00:00
return g_hash_table_get_values ( self - > calls ) ;
2020-03-17 15:29:43 +00:00
}
2021-01-11 23:50:28 +00:00
/**
* calls_manager_hang_up_all_calls :
* @ self : a # CallsManager
*
* Hangs up on every call known to @ self .
*/
void
calls_manager_hang_up_all_calls ( CallsManager * self )
{
2022-02-01 05:08:12 +00:00
GListModel * origins ;
2021-01-11 23:50:28 +00:00
GList * node ;
CallsCall * call ;
2022-02-01 05:08:12 +00:00
uint n_items ;
2021-01-11 23:50:28 +00:00
g_return_if_fail ( CALLS_IS_MANAGER ( self ) ) ;
2022-02-01 05:08:12 +00:00
origins = G_LIST_MODEL ( self - > origins ) ;
n_items = g_list_model_get_n_items ( origins ) ;
2021-01-11 23:50:28 +00:00
2022-02-01 05:08:12 +00:00
for ( uint i = 0 ; i < n_items ; i + + ) {
g_autoptr ( CallsOrigin ) origin = g_list_model_get_item ( origins , i ) ;
g_autoptr ( GList ) calls = calls_origin_get_calls ( origin ) ;
for ( node = calls ; node ; node = node - > next ) {
2021-01-11 23:50:28 +00:00
call = node - > data ;
g_debug ( " Hanging up on call %s " , calls_call_get_name ( call ) ) ;
calls_call_hang_up ( call ) ;
}
2022-02-01 05:08:12 +00:00
}
2021-01-11 23:50:28 +00:00
2022-02-01 05:08:12 +00:00
g_debug ( " Hung up on all calls " ) ;
2021-01-11 23:50:28 +00:00
}
2021-04-13 09:36:58 +00:00
/**
* calls_manager_get_suitable_origins :
* @ self : The # CallsManager
* @ target : The target number / address
*
2022-01-26 07:24:09 +00:00
* Returns ( transfer none ) : A # GListModel of suitable origins as long
* as the protocol to be used for @ target is supported , % NULL otherwise
2021-04-13 09:36:58 +00:00
*/
GListModel *
calls_manager_get_suitable_origins ( CallsManager * self ,
const char * target )
2020-03-17 15:29:43 +00:00
{
2021-04-13 09:36:58 +00:00
const char * protocol ;
GListModel * model ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
g_return_val_if_fail ( target , NULL ) ;
2021-03-25 19:25:09 +00:00
2021-04-13 09:36:58 +00:00
protocol = get_protocol_from_address_with_fallback ( target ) ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
model = g_hash_table_lookup ( self - > origins_by_protocol , protocol ) ;
if ( model & & G_IS_LIST_MODEL ( model ) )
return model ;
2020-03-17 15:29:43 +00:00
2021-04-13 09:36:58 +00:00
return NULL ;
2020-03-17 15:29:43 +00:00
}
2021-04-16 17:12:27 +00:00
/**
* calls_manager_has_any_provider :
* @ self : The # CallsManager
*
* Returns : % TRUE if any provider is loaded , % FALSE otherwise
*/
gboolean
calls_manager_has_any_provider ( CallsManager * self )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , FALSE ) ;
return ! ! g_hash_table_size ( self - > providers ) ;
}
2021-06-09 12:19:27 +00:00
/**
* calls_manager_get_provider_names :
* @ self : The # CallsManager
* @ length : ( optional ) ( out ) : the length of the returned array
*
* Retrieves the names of all providers loaded by @ self , as an array .
*
* You should free the return value with g_free ( ) .
*
* Returns : ( array length = length ) ( transfer container ) : a
* % NULL - terminated array containing the names of providers .
*/
const char * *
calls_manager_get_provider_names ( CallsManager * self ,
guint * length )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
return ( const char * * ) g_hash_table_get_keys_as_array ( self - > providers , length ) ;
}
2021-07-03 23:41:35 +00:00
/**
* calls_manager_get_providers :
* @ self : A # CallsManager
*
* Get the currently loaded providers
*
* Returns : ( transfer container ) : A # GList of # CallsProvider .
* Use g_list_free ( ) when done using the list .
*/
GList *
calls_manager_get_providers ( CallsManager * self )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
return g_hash_table_get_values ( self - > providers ) ;
}
2021-09-23 07:32:37 +00:00
CallsSettings *
calls_manager_get_settings ( CallsManager * self )
{
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
return self - > settings ;
}
2022-02-10 09:12:16 +00:00
/**
* calls_manager_get_origin_by_id :
* @ self : The # CallsManager
* @ origin_id : The id to use for the lookup
*
* Returns : ( transfer none ) : The # CallsOrigin if found , % NULL otherwise
*/
CallsOrigin *
calls_manager_get_origin_by_id ( CallsManager * self ,
const char * origin_id )
{
uint n_origins ;
g_return_val_if_fail ( CALLS_IS_MANAGER ( self ) , NULL ) ;
2022-04-01 09:38:01 +00:00
/* TODO Turn this into a critical once https://gitlab.gnome.org/GNOME/calls/-/merge_requests/505 is in */
if ( origin_id & & * origin_id )
return NULL ;
2022-02-10 09:12:16 +00:00
n_origins = g_list_model_get_n_items ( G_LIST_MODEL ( self - > origins ) ) ;
for ( uint i = 0 ; i < n_origins ; i + + ) {
g_autoptr ( CallsOrigin ) origin =
g_list_model_get_item ( G_LIST_MODEL ( self - > origins ) , i ) ;
g_autofree char * id = calls_origin_get_id ( origin ) ;
if ( g_strcmp0 ( id , origin_id ) = = 0 )
return origin ;
}
return NULL ;
}