2018-05-17 13:16:51 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 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: Bob Ham <bob.ham@puri.sm>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "calls-call.h"
|
|
|
|
#include "calls-message-source.h"
|
|
|
|
#include "enum-types.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
2019-06-28 08:59:02 +00:00
|
|
|
#include <glib/gi18n.h>
|
|
|
|
|
2018-05-17 13:16:51 +00:00
|
|
|
|
|
|
|
void
|
2018-05-23 08:52:58 +00:00
|
|
|
calls_call_state_to_string (GString *string,
|
|
|
|
CallsCallState state)
|
2018-05-17 13:16:51 +00:00
|
|
|
{
|
|
|
|
GEnumClass *klass;
|
|
|
|
GEnumValue *value;
|
|
|
|
|
|
|
|
klass = g_type_class_ref (CALLS_TYPE_CALL_STATE);
|
2019-07-01 14:02:08 +00:00
|
|
|
|
2018-05-17 13:16:51 +00:00
|
|
|
value = g_enum_get_value (klass, (gint)state);
|
2019-07-01 14:02:08 +00:00
|
|
|
if (!value)
|
|
|
|
{
|
|
|
|
return g_string_printf (string,
|
|
|
|
"Unknown call state (%d)",
|
|
|
|
(gint)state);
|
|
|
|
}
|
2018-05-17 13:16:51 +00:00
|
|
|
|
|
|
|
g_string_assign (string, value->value_nick);
|
|
|
|
string->str[0] = g_ascii_toupper (string->str[0]);
|
|
|
|
|
|
|
|
g_type_class_unref (klass);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
calls_call_state_parse_nick (CallsCallState *state,
|
|
|
|
const gchar *nick)
|
|
|
|
{
|
|
|
|
GEnumClass *klass;
|
|
|
|
GEnumValue *value;
|
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
g_return_val_if_fail (state != NULL, FALSE);
|
|
|
|
g_return_val_if_fail (nick != NULL, FALSE);
|
|
|
|
|
|
|
|
klass = g_type_class_ref (CALLS_TYPE_CALL_STATE);
|
|
|
|
value = g_enum_get_value_by_nick (klass, nick);
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
*state = (CallsCallState) value->value;
|
|
|
|
ret = TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_type_class_unref (klass);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:calls-call
|
|
|
|
* @short_description: A call.
|
|
|
|
* @Title: CallsCall
|
2018-05-23 08:52:58 +00:00
|
|
|
*
|
|
|
|
* This is the interface to a call. It has a number, name and a
|
|
|
|
* state. Only the state changes after creation. If the state is
|
|
|
|
* #CALL_CALL_STATE_INCOMING, the call can be answered with #answer.
|
|
|
|
* The call can also be hung up at any time with #hang_up.
|
|
|
|
*
|
|
|
|
* DTMF tones can be played the call using #tone_start and
|
|
|
|
* #tone_stop. Valid characters for the key are 0-9, '*', '#', 'A',
|
|
|
|
* 'B', 'C' and 'D'.
|
2018-05-17 13:16:51 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
G_DEFINE_INTERFACE (CallsCall, calls_call, CALLS_TYPE_MESSAGE_SOURCE);
|
|
|
|
|
|
|
|
enum {
|
|
|
|
SIGNAL_STATE_CHANGED,
|
|
|
|
SIGNAL_LAST_SIGNAL,
|
|
|
|
};
|
|
|
|
static guint signals [SIGNAL_LAST_SIGNAL];
|
|
|
|
|
2018-05-23 08:52:58 +00:00
|
|
|
|
2018-05-17 13:16:51 +00:00
|
|
|
static void
|
|
|
|
calls_call_default_init (CallsCallInterface *iface)
|
|
|
|
{
|
2018-11-09 12:38:04 +00:00
|
|
|
GType arg_types[2] =
|
|
|
|
{
|
|
|
|
CALLS_TYPE_CALL_STATE,
|
|
|
|
CALLS_TYPE_CALL_STATE
|
|
|
|
};
|
2018-05-17 13:16:51 +00:00
|
|
|
|
2020-02-22 23:55:45 +00:00
|
|
|
g_object_interface_install_property (
|
|
|
|
iface,
|
2019-06-28 08:59:02 +00:00
|
|
|
g_param_spec_boolean ("inbound",
|
2020-05-29 05:38:11 +00:00
|
|
|
"Inbound",
|
|
|
|
"Whether the call is inbound",
|
2019-06-28 08:59:02 +00:00
|
|
|
FALSE,
|
2020-02-22 23:55:45 +00:00
|
|
|
G_PARAM_READABLE));
|
|
|
|
|
|
|
|
g_object_interface_install_property (
|
|
|
|
iface,
|
|
|
|
g_param_spec_string ("number",
|
2020-05-29 05:38:11 +00:00
|
|
|
"Number",
|
|
|
|
"The number the call is connected to if known",
|
|
|
|
NULL,
|
|
|
|
G_PARAM_READABLE));
|
2020-02-22 23:55:45 +00:00
|
|
|
|
|
|
|
g_object_interface_install_property (
|
|
|
|
iface,
|
|
|
|
g_param_spec_string ("name",
|
2020-05-29 05:38:11 +00:00
|
|
|
"Name",
|
|
|
|
"The name of the party the call is connected to, if the network provides it",
|
|
|
|
NULL,
|
|
|
|
G_PARAM_READABLE));
|
2020-02-22 23:55:45 +00:00
|
|
|
|
|
|
|
g_object_interface_install_property (
|
|
|
|
iface,
|
|
|
|
g_param_spec_enum ("state",
|
2020-05-29 05:38:11 +00:00
|
|
|
"State",
|
|
|
|
"The current state of the call",
|
|
|
|
CALLS_TYPE_CALL_STATE,
|
|
|
|
CALLS_CALL_STATE_ACTIVE,
|
|
|
|
G_PARAM_READABLE));
|
2019-06-28 08:59:02 +00:00
|
|
|
|
2018-05-23 08:52:58 +00:00
|
|
|
/**
|
|
|
|
* CallsCall::state-changed:
|
|
|
|
* @self: The #CallsCall instance.
|
2018-11-09 12:38:04 +00:00
|
|
|
* @new_state: The new state of the call.
|
|
|
|
* @old_state: The old state of the call.
|
2018-05-23 08:52:58 +00:00
|
|
|
*
|
|
|
|
* This signal is emitted when the state of the call changes, for
|
|
|
|
* example when it's answered or when the call is disconnected.
|
|
|
|
*/
|
2018-05-17 13:16:51 +00:00
|
|
|
signals[SIGNAL_STATE_CHANGED] =
|
|
|
|
g_signal_newv ("state-changed",
|
|
|
|
G_TYPE_FROM_INTERFACE (iface),
|
|
|
|
G_SIGNAL_RUN_LAST,
|
|
|
|
NULL, NULL, NULL, NULL,
|
|
|
|
G_TYPE_NONE,
|
2018-11-09 12:38:04 +00:00
|
|
|
2, arg_types);
|
2018-05-17 13:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-22 23:55:45 +00:00
|
|
|
#define DEFINE_CALL_GETTER(prop,rettype,errval) \
|
|
|
|
CALLS_DEFINE_IFACE_GETTER(call, Call, CALL, prop, rettype, errval)
|
2018-05-17 13:16:51 +00:00
|
|
|
|
|
|
|
#define DEFINE_CALL_FUNC_VOID(function) \
|
|
|
|
CALLS_DEFINE_IFACE_FUNC_VOID(call, Call, CALL, function)
|
|
|
|
|
|
|
|
|
2018-05-23 08:52:58 +00:00
|
|
|
/**
|
|
|
|
* calls_call_get_number:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
|
|
|
* Get the number the call is connected to. It is possible that this
|
|
|
|
* could return NULL if the number is not known, for example if an
|
|
|
|
* incoming PTSN call has no caller ID information.
|
|
|
|
*
|
|
|
|
* Returns: the number, or NULL
|
|
|
|
*/
|
2020-02-22 23:55:45 +00:00
|
|
|
DEFINE_CALL_GETTER(number, const gchar *, NULL);
|
2018-05-23 08:52:58 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* calls_call_get_name:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
|
|
|
* Get the name of the party the call is connected to, if the network
|
|
|
|
* provides it.
|
|
|
|
*
|
|
|
|
* Returns: the number, or NULL
|
|
|
|
*/
|
2020-02-22 23:55:45 +00:00
|
|
|
DEFINE_CALL_GETTER(name, const gchar *, NULL);
|
2018-05-17 13:16:51 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* calls_call_get_state:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
2018-05-23 08:52:58 +00:00
|
|
|
* Get the current state of the call.
|
2018-05-17 13:16:51 +00:00
|
|
|
*
|
|
|
|
* Returns: the state
|
|
|
|
*/
|
2020-02-22 23:55:45 +00:00
|
|
|
DEFINE_CALL_GETTER(state, CallsCallState, ((CallsCallState)0));
|
2018-05-17 13:16:51 +00:00
|
|
|
|
2018-05-23 08:52:58 +00:00
|
|
|
/**
|
|
|
|
* calls_call_answer:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
|
|
|
* If the call is incoming, answer it.
|
|
|
|
*
|
|
|
|
*/
|
2018-05-17 13:16:51 +00:00
|
|
|
DEFINE_CALL_FUNC_VOID(answer);
|
2018-05-23 08:52:58 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* calls_call_hang_up:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
|
|
|
* Hang up the call.
|
|
|
|
*
|
|
|
|
*/
|
2018-05-17 13:16:51 +00:00
|
|
|
DEFINE_CALL_FUNC_VOID(hang_up);
|
|
|
|
|
|
|
|
|
2019-07-22 10:52:42 +00:00
|
|
|
/**
|
|
|
|
* calls_call_get_inbound:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
|
|
|
* Get the direction of the call.
|
|
|
|
*
|
|
|
|
* Returns: TRUE if inbound, FALSE if outbound.
|
|
|
|
*/
|
2020-02-22 23:55:45 +00:00
|
|
|
DEFINE_CALL_GETTER(inbound, gboolean, FALSE);
|
2019-07-22 10:52:42 +00:00
|
|
|
|
2018-05-17 13:16:51 +00:00
|
|
|
static inline gboolean
|
|
|
|
tone_key_is_valid (gchar key)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
(key >= '0' && key <= '9')
|
|
|
|
|| (key >= 'A' && key <= 'D')
|
|
|
|
|| key == '*'
|
|
|
|
|| key == '#';
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-23 08:52:58 +00:00
|
|
|
/**
|
|
|
|
* calls_call_tone_start:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
* @key: which tone to start
|
|
|
|
*
|
|
|
|
* Start playing a DTMF tone for the specified key. Implementations
|
|
|
|
* will stop playing the tone either after an implementation-specific
|
|
|
|
* timeout, or after #calls_call_tone_stop is called with the same
|
|
|
|
* value for @key.
|
|
|
|
*
|
|
|
|
*/
|
2018-05-31 10:20:17 +00:00
|
|
|
void
|
|
|
|
calls_call_tone_start (CallsCall *self,
|
|
|
|
gchar key)
|
|
|
|
{
|
|
|
|
CallsCallInterface *iface;
|
|
|
|
|
|
|
|
g_return_if_fail (CALLS_IS_CALL (self));
|
|
|
|
g_return_if_fail (tone_key_is_valid (key));
|
|
|
|
|
|
|
|
iface = CALLS_CALL_GET_IFACE (self);
|
|
|
|
g_return_if_fail (iface->tone_start != NULL);
|
|
|
|
|
|
|
|
iface->tone_start (self, key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calls_call_tone_stoppable:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
*
|
|
|
|
* Determine whether tones for this call can be stopped by calling
|
|
|
|
* #calls_call_tone_stop. Some implementations will only allow
|
|
|
|
* fixed-length tones to be played. In that case, this function
|
|
|
|
* should return FALSE.
|
|
|
|
*
|
|
|
|
* Returns: whether calls to #calls_call_tone_stop will do anything
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
calls_call_tone_stoppable (CallsCall *self)
|
|
|
|
{
|
|
|
|
CallsCallInterface *iface;
|
|
|
|
|
|
|
|
g_return_val_if_fail (CALLS_IS_CALL (self), FALSE);
|
|
|
|
|
|
|
|
iface = CALLS_CALL_GET_IFACE (self);
|
|
|
|
|
|
|
|
return iface->tone_stop != NULL;
|
|
|
|
}
|
2018-05-23 08:52:58 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* calls_call_tone_stop:
|
|
|
|
* @self: a #CallsCall
|
|
|
|
* @key: which tone to stop
|
|
|
|
*
|
2018-05-31 10:20:17 +00:00
|
|
|
* Stop playing a DTMF tone previously started with
|
|
|
|
* #calls_call_tone_start.
|
2018-05-23 08:52:58 +00:00
|
|
|
*
|
|
|
|
*/
|
2018-05-31 10:20:17 +00:00
|
|
|
void
|
|
|
|
calls_call_tone_stop (CallsCall *self,
|
|
|
|
gchar key)
|
|
|
|
{
|
|
|
|
CallsCallInterface *iface;
|
|
|
|
|
|
|
|
g_return_if_fail (CALLS_IS_CALL (self));
|
|
|
|
g_return_if_fail (tone_key_is_valid (key));
|
|
|
|
|
|
|
|
iface = CALLS_CALL_GET_IFACE (self);
|
|
|
|
if (!iface->tone_stop)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
iface->tone_stop (self, key);
|
|
|
|
}
|