/*
* Copyright (C) 2018, 2019, 2022 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 .
*
* Author: Adrien Plazas
* Evangelos Ribeiro Tzaras
*
* SPDX-License-Identifier: GPL-3.0-or-later
*
*/
#define G_LOG_DOMAIN "CallsHistoryBox"
#include "calls-history-box.h"
#include "calls-call-record.h"
#include "calls-call-record-row.h"
#include "calls-util.h"
#include
#include
#define CALLS_HISTORY_SIZE_INITIAL 75
#define CALLS_HISTORY_SIZE_INCREMENTS 50
#define CALLS_HISTORY_RESET_SIZE_POSITION_THRESHOLD 500
#define CALLS_HISTORY_INCREASE_N_PAGES_THRESHOLD 2
struct _CallsHistoryBox {
AdwBin parent_instance;
GtkStack *stack;
GtkListBox *history;
GtkScrolledWindow *scrolled_window;
GtkAdjustment *scroll_adjustment;
GListModel *model;
GtkSliceListModel *slice_model;
gsize n_items;
gulong model_changed_handler_id;
};
G_DEFINE_TYPE (CallsHistoryBox, calls_history_box, ADW_TYPE_BIN);
enum {
PROP_0,
PROP_MODEL,
PROP_LAST_PROP,
};
static GParamSpec *props[PROP_LAST_PROP];
static void
on_model_changed (GListModel *model,
guint position,
guint removed,
guint added,
CallsHistoryBox *self)
{
gchar *child_name;
self->n_items = self->n_items + added - removed;
if (self->n_items == 0)
child_name = "empty";
else
child_name = "history";
gtk_stack_set_visible_child_name (self->stack, child_name);
}
static void
delete_call_cb (CallsCallRecord *record,
CallsHistoryBox *self)
{
guint position;
guint id;
gboolean ok;
g_return_if_fail (CALLS_IS_CALL_RECORD (record));
ok = calls_find_in_model (self->model,
record,
&position);
g_object_get (G_OBJECT (record),
"id",
&id,
NULL);
if (!ok) {
g_warning ("Could not find record with id %u in model",
id);
return;
}
g_list_store_remove ((GListStore *) self->model, position);
}
static GtkWidget *
create_row_cb (CallsCallRecord *record,
CallsHistoryBox *self)
{
GtkWidget *row_widget;
row_widget = GTK_WIDGET (calls_call_record_row_new (record));
g_signal_connect (record,
"call-delete",
G_CALLBACK (delete_call_cb),
self);
return row_widget;
}
static void
on_adjustment_position_changed (GtkAdjustment *adjustment,
CallsHistoryBox *self)
{
double position;
double upper_limit;
double page_size;
guint old_size;
g_assert (CALLS_IS_HISTORY_BOX (self));
position = gtk_adjustment_get_value (adjustment);
page_size = gtk_adjustment_get_page_size (adjustment);
old_size = gtk_slice_list_model_get_size (self->slice_model);
if (position < CALLS_HISTORY_RESET_SIZE_POSITION_THRESHOLD) {
if (old_size == CALLS_HISTORY_SIZE_INITIAL)
return;
g_debug ("Resetting to initial size: %u",
CALLS_HISTORY_SIZE_INITIAL);
gtk_slice_list_model_set_size (self->slice_model, CALLS_HISTORY_SIZE_INITIAL);
return;
}
upper_limit = gtk_adjustment_get_upper (adjustment);
if (position > upper_limit - CALLS_HISTORY_INCREASE_N_PAGES_THRESHOLD * page_size) {
guint new_size = old_size + CALLS_HISTORY_SIZE_INCREMENTS;
new_size = MIN (new_size, self->n_items);
if (old_size == new_size)
return;
g_debug ("Increasing history slice from %u to %u",
old_size, new_size);
gtk_slice_list_model_set_size (self->slice_model, new_size);
}
}
static void
set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
CallsHistoryBox *self = CALLS_HISTORY_BOX (object);
switch (property_id) {
case PROP_MODEL:
g_set_object (&self->model,
G_LIST_MODEL (g_value_get_object (value)));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
break;
}
}
static void
constructed (GObject *object)
{
CallsHistoryBox *self = CALLS_HISTORY_BOX (object);
g_assert (self->model != NULL);
G_OBJECT_CLASS (calls_history_box_parent_class)->constructed (object);
self->slice_model = gtk_slice_list_model_new (self->model,
0,
CALLS_HISTORY_SIZE_INITIAL);
self->model_changed_handler_id =
g_signal_connect (self->model,
"items-changed",
G_CALLBACK (on_model_changed),
self);
g_assert (self->model_changed_handler_id != 0);
gtk_list_box_bind_model (self->history,
G_LIST_MODEL (self->slice_model),
(GtkListBoxCreateWidgetFunc) create_row_cb,
self,
NULL);
on_model_changed (self->model, 0, 0, g_list_model_get_n_items (self->model), self);
}
static void
dispose (GObject *object)
{
CallsHistoryBox *self = CALLS_HISTORY_BOX (object);
GtkWidget *stack = GTK_WIDGET (self->stack);
g_clear_signal_handler (&self->model_changed_handler_id, self->model);
g_clear_object (&self->slice_model);
g_clear_object (&self->model);
g_clear_pointer (&stack, gtk_widget_unparent);
G_OBJECT_CLASS (calls_history_box_parent_class)->dispose (object);
}
static void
calls_history_box_class_init (CallsHistoryBoxClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
object_class->set_property = set_property;
object_class->constructed = constructed;
object_class->dispose = dispose;
props[PROP_MODEL] =
g_param_spec_object ("model",
"model",
"The data store containing call records",
G_TYPE_LIST_MODEL,
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY);
g_object_class_install_properties (object_class, PROP_LAST_PROP, props);
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/Calls/ui/history-box.ui");
gtk_widget_class_bind_template_child (widget_class, CallsHistoryBox, stack);
gtk_widget_class_bind_template_child (widget_class, CallsHistoryBox, history);
gtk_widget_class_bind_template_child (widget_class, CallsHistoryBox, scrolled_window);
}
static void
calls_history_box_init (CallsHistoryBox *self)
{
gtk_widget_init_template (GTK_WIDGET (self));
self->scroll_adjustment = gtk_scrolled_window_get_vadjustment (self->scrolled_window);
g_signal_connect (self->scroll_adjustment,
"value-changed",
G_CALLBACK (on_adjustment_position_changed),
self);
}
CallsHistoryBox *
calls_history_box_new (GListModel *model)
{
return g_object_new (CALLS_TYPE_HISTORY_BOX,
"model", model,
NULL);
}