mirror of
https://gitlab.gnome.org/GNOME/calls.git
synced 2025-01-07 12:25:31 +00:00
992a243de6
Fixes #277
577 lines
17 KiB
C
577 lines
17 KiB
C
/*
|
|
* Copyright (C) 2021 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: Evangelos Ribeiro Tzaras <evangelos.tzaras@puri.sm>
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*
|
|
*/
|
|
|
|
#define G_LOG_DOMAIN "CallsSipAccountWidget"
|
|
|
|
#include "calls-sip-account-widget.h"
|
|
#include "calls-sip-provider.h"
|
|
#include "calls-sip-origin.h"
|
|
|
|
/**
|
|
* Section:calls-sip-account-widget
|
|
* short_description: A #GtkWidget to edit or add SIP accounts
|
|
* @Title: CallsSipAccountWidget
|
|
*
|
|
* This #GtkWidget allows the user to add a new or edit an existing SIP account.
|
|
*/
|
|
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_PROVIDER,
|
|
PROP_ORIGIN,
|
|
PROP_LAST_PROP
|
|
};
|
|
static GParamSpec *props[PROP_LAST_PROP];
|
|
|
|
|
|
struct _CallsSipAccountWidget {
|
|
GtkBox parent;
|
|
|
|
/* Header bar */
|
|
GtkWidget *header_add;
|
|
GtkSpinner *spinner_add;
|
|
GtkWidget *header_edit;
|
|
GtkSpinner *spinner_edit;
|
|
GtkWidget *login_btn;
|
|
GtkWidget *apply_btn;
|
|
GtkWidget *delete_btn;
|
|
|
|
/* widgets for editing account credentials */
|
|
GtkEntry *host;
|
|
GtkEntry *display_name;
|
|
GtkEntry *user;
|
|
GtkEntry *password;
|
|
GtkEntry *port;
|
|
char *last_port;
|
|
HdyComboRow *protocol;
|
|
GListStore *protocols_store; /* bound model for protocol HdyComboRow */
|
|
GtkSwitch *tel_switch;
|
|
|
|
|
|
/* properties */
|
|
CallsSipProvider *provider;
|
|
CallsSipOrigin *origin; /* nullable to add a new account */
|
|
|
|
/* misc */
|
|
gboolean connecting;
|
|
gboolean port_self_change;
|
|
};
|
|
|
|
G_DEFINE_TYPE (CallsSipAccountWidget, calls_sip_account_widget, GTK_TYPE_BOX)
|
|
|
|
|
|
static gboolean
|
|
is_form_valid (CallsSipAccountWidget *self)
|
|
{
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
/* TODO perform some sanity checks */
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
is_form_filled (CallsSipAccountWidget *self)
|
|
{
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
return
|
|
g_strcmp0 (gtk_entry_get_text (GTK_ENTRY (self->host)), "") != 0 &&
|
|
g_strcmp0 (gtk_entry_get_text (GTK_ENTRY (self->user)), "") != 0 &&
|
|
g_strcmp0 (gtk_entry_get_text (GTK_ENTRY (self->password)), "") != 0 &&
|
|
g_strcmp0 (gtk_entry_get_text (GTK_ENTRY (self->port)), "") != 0;
|
|
}
|
|
|
|
|
|
static void
|
|
on_user_changed (CallsSipAccountWidget *self)
|
|
{
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
gtk_widget_set_sensitive (self->login_btn,
|
|
is_form_filled (self) &&
|
|
is_form_valid (self));
|
|
|
|
gtk_widget_set_sensitive (self->apply_btn,
|
|
is_form_filled (self) &&
|
|
is_form_valid (self));
|
|
}
|
|
|
|
|
|
/*
|
|
* Stop "insert-text" signal emission if any undesired port
|
|
* value occurs
|
|
*/
|
|
static void
|
|
on_port_entry_insert_text (CallsSipAccountWidget *self,
|
|
char *new_text,
|
|
int new_text_length,
|
|
gpointer position,
|
|
GtkEntry *entry)
|
|
{
|
|
size_t digit_end, len;
|
|
int *pos;
|
|
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
g_assert (GTK_IS_ENTRY (entry));
|
|
|
|
if (!new_text || !*new_text || self->port_self_change)
|
|
return;
|
|
|
|
pos = (int *)position;
|
|
g_object_set_data (G_OBJECT (entry), "old-pos", GINT_TO_POINTER (*pos));
|
|
|
|
if (new_text_length == -1)
|
|
len = strlen (new_text);
|
|
else
|
|
len = new_text_length;
|
|
|
|
digit_end = strspn (new_text, "1234567890");
|
|
|
|
/* If user inserted something other than a digit,
|
|
* stop inserting the text and warn the user.
|
|
*/
|
|
if (digit_end != len) {
|
|
g_signal_stop_emission_by_name (entry, "insert-text");
|
|
gtk_widget_error_bell (GTK_WIDGET (entry));
|
|
} else {
|
|
g_free (self->last_port);
|
|
self->last_port = g_strdup (gtk_entry_get_text (entry));
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean
|
|
update_port_cursor_position (GtkEntry *entry)
|
|
{
|
|
int pos;
|
|
|
|
pos = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (entry), "old-pos"));
|
|
gtk_editable_set_position (GTK_EDITABLE (entry), pos);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
|
|
static int
|
|
get_port (CallsSipAccountWidget *self)
|
|
{
|
|
const char *text;
|
|
int port = 0;
|
|
|
|
text = gtk_entry_get_text (self->port);
|
|
port = (int)g_ascii_strtod (text, NULL);
|
|
|
|
return port;
|
|
}
|
|
|
|
|
|
static void
|
|
on_port_entry_after_insert_text (CallsSipAccountWidget *self,
|
|
char *new_text,
|
|
int new_text_length,
|
|
gpointer position,
|
|
GtkEntry *entry)
|
|
{
|
|
int port = get_port (self);
|
|
|
|
/* Reset to the old value if new port number is invalid */
|
|
if ((port < 0 || port > 65535) && self->last_port) {
|
|
self->port_self_change = TRUE;
|
|
gtk_entry_set_text (entry, self->last_port);
|
|
g_idle_add (G_SOURCE_FUNC (update_port_cursor_position), entry);
|
|
gtk_widget_error_bell (GTK_WIDGET (entry));
|
|
self->port_self_change = FALSE;
|
|
}
|
|
}
|
|
|
|
static void
|
|
update_header (CallsSipAccountWidget *self)
|
|
{
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
if (self->origin) {
|
|
gtk_widget_show (self->header_edit);
|
|
gtk_widget_hide (self->header_add);
|
|
|
|
} else {
|
|
gtk_widget_show (self->header_add);
|
|
gtk_widget_hide (self->header_edit);
|
|
}
|
|
|
|
if (self->connecting) {
|
|
gtk_spinner_start (self->spinner_add);
|
|
gtk_spinner_start (self->spinner_edit);
|
|
} else {
|
|
gtk_spinner_stop (self->spinner_add);
|
|
gtk_spinner_stop (self->spinner_edit);
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean
|
|
find_protocol (CallsSipAccountWidget *self,
|
|
const char *protocol,
|
|
guint *index)
|
|
{
|
|
guint len;
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
len = g_list_model_get_n_items (G_LIST_MODEL (self->protocols_store));
|
|
for (guint i = 0; i < len; i++) {
|
|
g_autoptr (HdyValueObject) obj =
|
|
g_list_model_get_item (G_LIST_MODEL (self->protocols_store), i);
|
|
const char *prot = hdy_value_object_get_string (obj);
|
|
|
|
if (g_strcmp0 (protocol, prot) == 0) {
|
|
if (index)
|
|
*index = i;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
g_debug ("Could not find protocol '%s'", protocol);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void
|
|
clear_form (CallsSipAccountWidget *self)
|
|
{
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
gtk_entry_set_text (self->host, "");
|
|
gtk_entry_set_text (self->display_name, "");
|
|
gtk_entry_set_text (self->user, "");
|
|
gtk_entry_set_text (self->password, "");
|
|
gtk_entry_set_text (self->port, "0");
|
|
hdy_combo_row_set_selected_index (self->protocol, 0);
|
|
gtk_switch_set_state (self->tel_switch, FALSE);
|
|
|
|
self->origin = NULL;
|
|
|
|
update_header (self);
|
|
|
|
if (gtk_widget_get_can_focus (GTK_WIDGET (self->host)))
|
|
gtk_widget_grab_focus (GTK_WIDGET (self->host));
|
|
}
|
|
|
|
|
|
static void
|
|
edit_form (CallsSipAccountWidget *self,
|
|
CallsSipOrigin *origin)
|
|
{
|
|
g_autofree char *host = NULL;
|
|
g_autofree char *display_name = NULL;
|
|
g_autofree char *user = NULL;
|
|
g_autofree char *password = NULL;
|
|
g_autofree char *port_str = NULL;
|
|
g_autofree char *protocol = NULL;
|
|
gint port;
|
|
guint protocol_index;
|
|
gboolean can_tel;
|
|
|
|
g_assert (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
|
|
if (!origin) {
|
|
clear_form (self);
|
|
return;
|
|
}
|
|
|
|
g_assert (CALLS_IS_SIP_ORIGIN (origin));
|
|
|
|
self->origin = origin;
|
|
|
|
g_object_get (origin,
|
|
"host", &host,
|
|
"display-name", &display_name,
|
|
"user", &user,
|
|
"password", &password,
|
|
"port", &port,
|
|
"transport-protocol", &protocol,
|
|
"can-tel", &can_tel,
|
|
NULL);
|
|
|
|
port_str = g_strdup_printf ("%d", port);
|
|
|
|
/* The following should always succeed,
|
|
TODO inform user in the error case
|
|
related issue #275 https://source.puri.sm/Librem5/calls/-/issues/275
|
|
*/
|
|
if (!find_protocol (self, protocol, &protocol_index))
|
|
protocol_index = 0;
|
|
|
|
/* set UI elements */
|
|
gtk_entry_set_text (self->host, host);
|
|
gtk_entry_set_text (self->display_name, display_name ?: "");
|
|
gtk_entry_set_text (self->user, user);
|
|
gtk_entry_set_text (self->password, password);
|
|
gtk_entry_set_text (self->port, port_str);
|
|
hdy_combo_row_set_selected_index (self->protocol, protocol_index);
|
|
gtk_switch_set_state (self->tel_switch, can_tel);
|
|
|
|
gtk_widget_set_sensitive (self->apply_btn, FALSE);
|
|
|
|
update_header (self);
|
|
|
|
if (gtk_widget_get_can_focus (GTK_WIDGET (self->host)))
|
|
gtk_widget_grab_focus (GTK_WIDGET (self->host));
|
|
}
|
|
|
|
|
|
static const char *
|
|
get_selected_protocol (CallsSipAccountWidget *self)
|
|
{
|
|
g_autoptr (HdyValueObject) obj = NULL;
|
|
const char *protocol = NULL;
|
|
gint i;
|
|
|
|
if ((i = hdy_combo_row_get_selected_index (self->protocol)) != -1) {
|
|
obj = g_list_model_get_item (G_LIST_MODEL (self->protocols_store), i);
|
|
protocol = hdy_value_object_get_string (obj);
|
|
}
|
|
return protocol;
|
|
}
|
|
|
|
|
|
static void
|
|
on_login_clicked (CallsSipAccountWidget *self)
|
|
{
|
|
CallsSipOrigin *origin;
|
|
|
|
g_debug ("Logging into newly created account");
|
|
|
|
origin = calls_sip_provider_add_origin (self->provider,
|
|
gtk_entry_get_text (GTK_ENTRY (self->host)),
|
|
gtk_entry_get_text (GTK_ENTRY (self->user)),
|
|
gtk_entry_get_text (GTK_ENTRY (self->password)),
|
|
gtk_entry_get_text (GTK_ENTRY (self->display_name)),
|
|
get_selected_protocol (self),
|
|
get_port (self),
|
|
TRUE);
|
|
|
|
self->origin = origin;
|
|
update_header (self);
|
|
g_signal_emit_by_name (self->provider, "widget-edit-done");
|
|
}
|
|
|
|
|
|
static void
|
|
on_delete_clicked (CallsSipAccountWidget *self)
|
|
{
|
|
g_debug ("Deleting account");
|
|
|
|
calls_sip_provider_remove_origin (self->provider, self->origin);
|
|
self->origin = NULL;
|
|
|
|
update_header (self);
|
|
g_signal_emit_by_name (self->provider, "widget-edit-done");
|
|
}
|
|
|
|
|
|
static void
|
|
on_apply_clicked (CallsSipAccountWidget *self)
|
|
{
|
|
g_debug ("Applying changes to the account");
|
|
|
|
calls_sip_origin_set_credentials (self->origin,
|
|
gtk_entry_get_text (self->host),
|
|
gtk_entry_get_text (self->user),
|
|
gtk_entry_get_text (self->password),
|
|
gtk_entry_get_text (self->display_name),
|
|
get_selected_protocol (self),
|
|
get_port (self),
|
|
gtk_switch_get_state (self->tel_switch),
|
|
TRUE);
|
|
|
|
update_header (self);
|
|
calls_sip_provider_save_accounts_to_disk (self->provider);
|
|
g_signal_emit_by_name (self->provider, "widget-edit-done");
|
|
}
|
|
|
|
|
|
static void
|
|
calls_sip_account_widget_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CallsSipAccountWidget *self = CALLS_SIP_ACCOUNT_WIDGET (object);
|
|
|
|
switch (property_id) {
|
|
case PROP_PROVIDER:
|
|
self->provider = g_value_get_object (value);
|
|
break;
|
|
|
|
case PROP_ORIGIN:
|
|
calls_sip_account_widget_set_origin (self, g_value_get_object (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
calls_sip_account_widget_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CallsSipAccountWidget *self = CALLS_SIP_ACCOUNT_WIDGET (object);
|
|
|
|
switch (property_id) {
|
|
case PROP_ORIGIN:
|
|
g_value_set_object (value, calls_sip_account_widget_get_origin (self));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
calls_sip_account_widget_dispose (GObject *object)
|
|
{
|
|
CallsSipAccountWidget *self = CALLS_SIP_ACCOUNT_WIDGET (object);
|
|
|
|
g_clear_pointer (&self->last_port, g_free);
|
|
g_clear_object (&self->protocols_store);
|
|
|
|
G_OBJECT_CLASS (calls_sip_account_widget_parent_class)->dispose (object);
|
|
}
|
|
|
|
|
|
static void
|
|
calls_sip_account_widget_class_init (CallsSipAccountWidgetClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
object_class->set_property = calls_sip_account_widget_set_property;
|
|
object_class->get_property = calls_sip_account_widget_get_property;
|
|
object_class->dispose = calls_sip_account_widget_dispose;
|
|
|
|
props[PROP_PROVIDER] =
|
|
g_param_spec_object ("provider",
|
|
"Provider",
|
|
"The SIP provider",
|
|
CALLS_TYPE_SIP_PROVIDER,
|
|
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
|
|
|
|
props[PROP_ORIGIN] =
|
|
g_param_spec_object ("origin",
|
|
"Origin",
|
|
"The origin to edit",
|
|
CALLS_TYPE_SIP_ORIGIN,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (object_class, PROP_LAST_PROP, props);
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/Calls/ui/sip-account-widget.ui");
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, header_add);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, spinner_add);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, header_edit);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, spinner_edit);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, login_btn);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, apply_btn);
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, host);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, display_name);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, user);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, password);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, port);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, protocol);
|
|
gtk_widget_class_bind_template_child (widget_class, CallsSipAccountWidget, tel_switch);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, on_login_clicked);
|
|
gtk_widget_class_bind_template_callback (widget_class, on_delete_clicked);
|
|
gtk_widget_class_bind_template_callback (widget_class, on_apply_clicked);
|
|
gtk_widget_class_bind_template_callback (widget_class, on_user_changed);
|
|
gtk_widget_class_bind_template_callback (widget_class, on_port_entry_insert_text);
|
|
gtk_widget_class_bind_template_callback (widget_class, on_port_entry_after_insert_text);
|
|
}
|
|
|
|
|
|
static void
|
|
calls_sip_account_widget_init (CallsSipAccountWidget *self)
|
|
{
|
|
HdyValueObject *obj;
|
|
|
|
gtk_widget_init_template (GTK_WIDGET (self));
|
|
|
|
self->protocols_store = g_list_store_new (HDY_TYPE_VALUE_OBJECT);
|
|
|
|
obj = hdy_value_object_new_string ("UDP");
|
|
g_list_store_insert (self->protocols_store, 0, obj);
|
|
g_clear_object (&obj);
|
|
|
|
obj = hdy_value_object_new_string ("TCP");
|
|
g_list_store_insert (self->protocols_store, 1, obj);
|
|
g_clear_object (&obj);
|
|
|
|
obj = hdy_value_object_new_string ("TLS");
|
|
g_list_store_insert (self->protocols_store, 2, obj);
|
|
g_clear_object (&obj);
|
|
|
|
hdy_combo_row_bind_name_model (self->protocol,
|
|
G_LIST_MODEL (self->protocols_store),
|
|
(HdyComboRowGetNameFunc) hdy_value_object_dup_string,
|
|
NULL, NULL);
|
|
}
|
|
|
|
|
|
CallsSipAccountWidget *
|
|
calls_sip_account_widget_new (CallsSipProvider *provider)
|
|
{
|
|
g_return_val_if_fail (CALLS_IS_SIP_PROVIDER (provider), NULL);
|
|
|
|
return g_object_new (CALLS_TYPE_SIP_ACCOUNT_WIDGET,
|
|
"provider", provider,
|
|
NULL);
|
|
}
|
|
|
|
|
|
CallsSipOrigin *
|
|
calls_sip_account_widget_get_origin (CallsSipAccountWidget *self)
|
|
{
|
|
g_return_val_if_fail (CALLS_IS_SIP_ACCOUNT_WIDGET (self), NULL);
|
|
|
|
return self->origin;
|
|
}
|
|
|
|
|
|
void
|
|
calls_sip_account_widget_set_origin (CallsSipAccountWidget *self,
|
|
CallsSipOrigin *origin)
|
|
{
|
|
g_return_if_fail (CALLS_IS_SIP_ACCOUNT_WIDGET (self));
|
|
g_return_if_fail (!origin || CALLS_IS_SIP_ORIGIN (origin));
|
|
|
|
edit_form (self, origin);
|
|
}
|