blob: 25ab3cf2d279cfe24f2f30a1b2d2adbbdc40dbe9 [file] [log] [blame]
/*
* Copyright (C) 2015 Savoir-Faire Linux Inc.
* Author: Stepan Salenikovich <stepan.salenikovich@savoirfairelinux.com>
*
* This program 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.
*
* This program 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 this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Additional permission under GNU GPL version 3 section 7:
*
* If you modify this program, or any covered work, by linking or
* combining it with the OpenSSL project's OpenSSL library (or a
* modified version of that library), containing parts covered by the
* terms of the OpenSSL or SSLeay licenses, Savoir-Faire Linux Inc.
* grants you additional permission to convey the resulting work.
* Corresponding Source for a non-source form of such a combination
* shall include the source code for the parts of OpenSSL used as well
* as that of the covered work.
*/
#include "contactsview.h"
#include <gtk/gtk.h>
#include "models/gtkqsortfiltertreemodel.h"
#include "models/gtkqtreemodel.h"
#include "models/activeitemproxymodel.h"
#include <categorizedcontactmodel.h>
#include <personmodel.h>
#include "utils/calling.h"
#include <memory>
#include "delegates/pixbufdelegate.h"
#include <contactmethod.h>
#include "defines.h"
#include "utils/models.h"
#include <categorizedbookmarkmodel.h>
#include <call.h>
#define COPY_DATA_KEY "copy_data"
struct _ContactsView
{
GtkScrolledWindow parent;
};
struct _ContactsViewClass
{
GtkScrolledWindowClass parent_class;
};
typedef struct _ContactsViewPrivate ContactsViewPrivate;
struct _ContactsViewPrivate
{
ActiveItemProxyModel *q_contact_model;
};
G_DEFINE_TYPE_WITH_PRIVATE(ContactsView, contacts_view, GTK_TYPE_SCROLLED_WINDOW);
#define CONTACTS_VIEW_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), CONTACTS_VIEW_TYPE, ContactsViewPrivate))
static void
render_contact_photo(G_GNUC_UNUSED GtkTreeViewColumn *tree_column,
GtkCellRenderer *cell,
GtkTreeModel *tree_model,
GtkTreeIter *iter,
G_GNUC_UNUSED gpointer data)
{
/* check if this is a top level item (category),
* or a bottom level item (contact method)
* in this case we don't want to show a photo */
GtkTreePath *path = gtk_tree_model_get_path(tree_model, iter);
int depth = gtk_tree_path_get_depth(path);
gtk_tree_path_free(path);
if (depth == 2) {
/* get person */
QModelIndex idx = gtk_q_sort_filter_tree_model_get_source_idx(GTK_Q_SORT_FILTER_TREE_MODEL(tree_model), iter);
if (idx.isValid()) {
QVariant var_c = idx.data(static_cast<int>(Person::Role::Object));
Person *c = var_c.value<Person *>();
/* get photo */
QVariant var_p = PixbufDelegate::instance()->contactPhoto(c, QSize(50, 50), false);
std::shared_ptr<GdkPixbuf> photo = var_p.value<std::shared_ptr<GdkPixbuf>>();
g_object_set(G_OBJECT(cell), "pixbuf", photo.get(), NULL);
return;
}
}
/* otherwise, make sure its an empty pixbuf */
g_object_set(G_OBJECT(cell), "pixbuf", NULL, NULL);
}
static void
render_name_and_contact_method(G_GNUC_UNUSED GtkTreeViewColumn *tree_column,
GtkCellRenderer *cell,
GtkTreeModel *tree_model,
GtkTreeIter *iter,
G_GNUC_UNUSED gpointer data)
{
/**
* If contact (person), show the name and the contact method (number)
* underneath; if multiple contact methods, then indicate as such
*
* Otherwise just display the category or contact method
*/
GtkTreePath *path = gtk_tree_model_get_path(tree_model, iter);
int depth = gtk_tree_path_get_depth(path);
gtk_tree_path_free(path);
gchar *text = NULL;
QModelIndex idx = gtk_q_sort_filter_tree_model_get_source_idx(GTK_Q_SORT_FILTER_TREE_MODEL(tree_model), iter);
if (idx.isValid()) {
QVariant var = idx.data(Qt::DisplayRole);
if (depth == 1) {
/* category */
text = g_strdup_printf("<b>%s</b>", var.value<QString>().toUtf8().constData());
} else if (depth == 2) {
/* contact, check for contact methods */
QVariant var_c = idx.data(static_cast<int>(Person::Role::Object));
if (var_c.isValid()) {
Person *c = var_c.value<Person *>();
switch (c->phoneNumbers().size()) {
case 0:
text = g_strdup_printf("%s\n", c->formattedName().toUtf8().constData());
break;
case 1:
{
QString number;
QVariant var_n = c->phoneNumbers().first()->roleData(Qt::DisplayRole);
if (var_n.isValid())
number = var_n.value<QString>();
text = g_strdup_printf("%s\n <span fgcolor=\"gray\">%s</span>",
c->formattedName().toUtf8().constData(),
number.toUtf8().constData());
break;
}
default:
/* more than one, for now don't show any of the contact methods */
text = g_strdup_printf("%s\n", c->formattedName().toUtf8().constData());
break;
}
} else {
/* should never happen since depth 2 should always be a contact (person) */
text = g_strdup_printf("%s", var.value<QString>().toUtf8().constData());
}
} else {
/* contact method (or deeper??) */
text = g_strdup_printf("%s", var.value<QString>().toUtf8().constData());
}
}
g_object_set(G_OBJECT(cell), "markup", text, NULL);
g_free(text);
/* set the colour */
if ( depth == 1) {
/* nice blue taken from the ring logo */
GdkRGBA rgba = {0.2, 0.75294117647, 0.82745098039, 0.1};
g_object_set(G_OBJECT(cell), "cell-background-rgba", &rgba, NULL);
} else {
g_object_set(G_OBJECT(cell), "cell-background", NULL, NULL);
}
}
static void
expand_if_child(G_GNUC_UNUSED GtkTreeModel *tree_model,
GtkTreePath *path,
G_GNUC_UNUSED GtkTreeIter *iter,
GtkTreeView *treeview)
{
if (gtk_tree_path_get_depth(path) == 2)
gtk_tree_view_expand_to_path(treeview, path);
}
static void
activate_contact_item(GtkTreeView *tree_view,
GtkTreePath *path,
G_GNUC_UNUSED GtkTreeViewColumn *column,
G_GNUC_UNUSED gpointer user_data)
{
/* expand / contract row */
if (gtk_tree_view_row_expanded(tree_view, path))
gtk_tree_view_collapse_row(tree_view, path);
else
gtk_tree_view_expand_row(tree_view, path, FALSE);
GtkTreeModel *model = gtk_tree_view_get_model(tree_view);
/* get iter */
GtkTreeIter iter;
if (gtk_tree_model_get_iter(model, &iter, path)) {
QModelIndex idx = gtk_q_sort_filter_tree_model_get_source_idx(GTK_Q_SORT_FILTER_TREE_MODEL(model), &iter);
if (idx.isValid()) {
int depth = gtk_tree_path_get_depth(path);
switch (depth) {
case 0:
case 1:
/* category, nothing to do */
break;
case 2:
{
/* contact (person), use contact method if there is only one */
QVariant var_c = idx.data(static_cast<int>(Person::Role::Object));
if (var_c.isValid()) {
Person *c = var_c.value<Person *>();
if (c->phoneNumbers().size() == 1) {
/* call with contact method */
place_new_call(c->phoneNumbers().first());
}
}
break;
}
default:
{
/* contact method (or deeper) */
QVariant var_n = idx.data(static_cast<int>(ContactMethod::Role::Object));
if (var_n.isValid()) {
/* call with contat method */
place_new_call(var_n.value<ContactMethod *>());
}
break;
}
}
}
}
}
static void
copy_contact_info(GtkWidget *item, G_GNUC_UNUSED gpointer user_data)
{
gpointer data = g_object_get_data(G_OBJECT(item), COPY_DATA_KEY);
g_return_if_fail(data);
gchar* text = (gchar *)data;
GtkClipboard* clip = gtk_clipboard_get(GDK_SELECTION_CLIPBOARD);
gtk_clipboard_set_text(clip, text, -1);
}
static gboolean
contacts_popup_menu(G_GNUC_UNUSED GtkWidget *widget, GdkEventButton *event, GtkTreeView *treeview)
{
/* build popup menu when right clicking on contact item
* user should be able to copy the contact's name or "number".
* other functionality may be added later.
*/
/* check for right click */
if (event->button != BUTTON_RIGHT_CLICK || event->type != GDK_BUTTON_PRESS)
return FALSE;
/* we don't want a popup menu for categories for now, so everything deeper
* than one */
GtkTreeIter iter;
GtkTreeModel *model;
GtkTreeSelection *selection = gtk_tree_view_get_selection(treeview);
if (!gtk_tree_selection_get_selected(selection, &model, &iter))
return FALSE;
GtkTreePath *path = gtk_tree_model_get_path(model, &iter);
int depth = gtk_tree_path_get_depth(path);
gtk_tree_path_free(path);
if (depth < 2)
return FALSE;
/* deeper than a category, so create a menu */
GtkWidget *menu = gtk_menu_new();
QModelIndex idx = get_index_from_selection(selection);
/* if depth == 2, it is a contact, offer to copy name, and if only one
* contact method exists then also the "number",
* if depth > 2, then its a contact method, so only offer to copy the number
*/
if (depth == 2) {
QVariant var_c = idx.data(static_cast<int>(Person::Role::Object));
if (var_c.isValid()) {
Person *c = var_c.value<Person *>();
/* copy name */
gchar *name = g_strdup_printf("%s", c->formattedName().toUtf8().constData());
GtkWidget *item = gtk_menu_item_new_with_mnemonic("_Copy name");
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
g_object_set_data_full(G_OBJECT(item), COPY_DATA_KEY, name, (GDestroyNotify)g_free);
g_signal_connect(item,
"activate",
G_CALLBACK(copy_contact_info),
NULL);
/* copy number if there is only one */
if (c->phoneNumbers().size() == 1) {
gchar *number = g_strdup_printf("%s",c->phoneNumbers().first()->uri().toUtf8().constData());
GtkWidget *item = gtk_menu_item_new_with_mnemonic("_Copy number");
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
g_object_set_data_full(G_OBJECT(item), COPY_DATA_KEY, number, (GDestroyNotify)g_free);
g_signal_connect(item,
"activate",
G_CALLBACK(copy_contact_info),
NULL);
}
}
} else if (depth > 2) {
/* copy number */
QVariant var_n = idx.data(static_cast<int>(ContactMethod::Role::Object));
if (var_n.isValid()) {
ContactMethod *n = var_n.value<ContactMethod *>();
gchar *number = g_strdup_printf("%s",n->uri().toUtf8().constData());
GtkWidget *item = gtk_menu_item_new_with_mnemonic("_Copy number");
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
g_object_set_data_full(G_OBJECT(item), COPY_DATA_KEY, number, (GDestroyNotify)g_free);
g_signal_connect(item,
"activate",
G_CALLBACK(copy_contact_info),
NULL);
}
}
/* show menu */
gtk_widget_show_all(menu);
gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, event->button, event->time);
return TRUE; /* we handled the event */
}
static void
render_frequen_contact_photo(G_GNUC_UNUSED GtkTreeViewColumn *tree_column,
GtkCellRenderer *cell,
GtkTreeModel *tree_model,
GtkTreeIter *iter,
G_GNUC_UNUSED gpointer data)
{
/* check if this is a top level item (category),
* or a bottom level item (contact method)
* in this case we don't want to show a photo */
GtkTreePath *path = gtk_tree_model_get_path(tree_model, iter);
int depth = gtk_tree_path_get_depth(path);
gtk_tree_path_free(path);
if (depth == 2) {
/* get contact method */
QModelIndex idx = gtk_q_tree_model_get_source_idx(GTK_Q_TREE_MODEL(tree_model), iter);
if (idx.isValid()) {
auto n = CategorizedBookmarkModel::instance()->getNumber(idx);
/* get photo */
QVariant var_p = PixbufDelegate::instance()->callPhoto(n, QSize(50, 50), false);
std::shared_ptr<GdkPixbuf> photo = var_p.value<std::shared_ptr<GdkPixbuf>>();
g_object_set(G_OBJECT(cell), "pixbuf", photo.get(), NULL);
return;
}
/* set height to default */
g_object_set(G_OBJECT(cell), "height", -1, NULL);
} else {
/* set height to 0 to hide categories
* TODO: this is a hack because flat versions of this mdoel doesn't exist
* yet in LRC
*/
g_object_set(G_OBJECT(cell), "height", 0, NULL);
}
/* otherwise, make sure its an empty pixbuf */
g_object_set(G_OBJECT(cell), "pixbuf", NULL, NULL);
}
static void
render_frequent_name_and_contact_method(G_GNUC_UNUSED GtkTreeViewColumn *tree_column,
GtkCellRenderer *cell,
GtkTreeModel *tree_model,
GtkTreeIter *iter,
G_GNUC_UNUSED gpointer data)
{
GtkTreePath *path = gtk_tree_model_get_path(tree_model, iter);
int depth = gtk_tree_path_get_depth(path);
gtk_tree_path_free(path);
gchar *text = NULL;
if (depth == 2) {
/* get contact method */
QModelIndex idx = gtk_q_tree_model_get_source_idx(GTK_Q_TREE_MODEL(tree_model), iter);
if (idx.isValid()) {
/* get name and number */
QVariant c = idx.data(static_cast<int>(Call::Role::Name));
QVariant n = idx.data(static_cast<int>(Call::Role::Number));
text = g_strdup_printf("%s\n <span fgcolor=\"gray\">%s</span>",
c.value<QString>().toUtf8().constData(),
n.value<QString>().toUtf8().constData());
}
/* set height to default */
g_object_set(G_OBJECT(cell), "height", -1, NULL);
} else {
/* set height to 0 to hide categories
* TODO: this is a hack because flat versions of this mdoel doesn't exist
* yet in LRC
*/
g_object_set(G_OBJECT(cell), "height", 0, NULL);
}
g_object_set(G_OBJECT(cell), "markup", text, NULL);
g_free(text);
}
static void
activate_frequent_item(GtkTreeView *tree_view,
GtkTreePath *path,
G_GNUC_UNUSED GtkTreeViewColumn *column,
G_GNUC_UNUSED gpointer user_data)
{
GtkTreeModel *model = gtk_tree_view_get_model(tree_view);
GtkTreeIter iter;
if (gtk_tree_model_get_iter(model, &iter, path)) {
QModelIndex idx = gtk_q_tree_model_get_source_idx(GTK_Q_TREE_MODEL(model), &iter);
if (idx.isValid()) {
int depth = gtk_tree_path_get_depth(path);
if (depth == 2) {
QVariant var_n = idx.data(static_cast<int>(Call::Role::ContactMethod));
if (var_n.isValid())
place_new_call(var_n.value<ContactMethod *>());
}
}
}
}
static gboolean
frequent_popup_menu(GtkTreeView *treeview, GdkEventButton *event, G_GNUC_UNUSED gpointer user_data)
{
/* build popup menu when right clicking on contact item
* user should be able to copy the contact's name or "number".
* other functionality may be added later.
*/
/* check for right click */
if (event->button != BUTTON_RIGHT_CLICK || event->type != GDK_BUTTON_PRESS)
return FALSE;
/* we don't want a popup menu for categories for now, so everything deeper
* than one */
GtkTreeIter iter;
GtkTreeModel *model;
GtkTreeSelection *selection = gtk_tree_view_get_selection(treeview);
if (!gtk_tree_selection_get_selected(selection, &model, &iter))
return FALSE;
GtkTreePath *path = gtk_tree_model_get_path(model, &iter);
int depth = gtk_tree_path_get_depth(path);
gtk_tree_path_free(path);
if (depth != 2)
return FALSE;
/* deeper than a category, so create a menu */
GtkWidget *menu = gtk_menu_new();
QModelIndex idx = get_index_from_selection(selection);
/* get name and number */
QVariant c = idx.data(static_cast<int>(Call::Role::Name));
QVariant n = idx.data(static_cast<int>(Call::Role::Number));
/* copy name */
gchar *name = g_strdup_printf("%s", c.value<QString>().toUtf8().constData());
GtkWidget *item = gtk_menu_item_new_with_mnemonic("_Copy name");
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
g_object_set_data_full(G_OBJECT(item), COPY_DATA_KEY, name, (GDestroyNotify)g_free);
g_signal_connect(item,
"activate",
G_CALLBACK(copy_contact_info),
NULL);
gchar *number = g_strdup_printf("%s", n.value<QString>().toUtf8().constData());
item = gtk_menu_item_new_with_mnemonic("_Copy number");
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
g_object_set_data_full(G_OBJECT(item), COPY_DATA_KEY, number, (GDestroyNotify)g_free);
g_signal_connect(item,
"activate",
G_CALLBACK(copy_contact_info),
NULL);
/* show menu */
gtk_widget_show_all(menu);
gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, event->button, event->time);
return TRUE; /* we handled the event */
}
static void
contacts_view_init(ContactsView *self)
{
ContactsViewPrivate *priv = CONTACTS_VIEW_GET_PRIVATE(self);
GtkWidget *vbox_main = gtk_box_new(GTK_ORIENTATION_VERTICAL, 10);
gtk_container_add(GTK_CONTAINER(self), vbox_main);
/* frequent contacts/numbers */
GtkWidget *label_frequent = gtk_label_new("Frequent Contacts");
gtk_box_pack_start(GTK_BOX(vbox_main), label_frequent, FALSE, TRUE, 0);
GtkWidget *treeview_frequent = gtk_tree_view_new();
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview_frequent), FALSE);
gtk_box_pack_start(GTK_BOX(vbox_main), treeview_frequent, FALSE, TRUE, 0);
/* no need to show the expander since it will always be expanded */
gtk_tree_view_set_show_expanders(GTK_TREE_VIEW(treeview_frequent), FALSE);
/* disable default search, we will handle it ourselves via LRC;
* otherwise the search steals input focus on key presses */
gtk_tree_view_set_enable_search(GTK_TREE_VIEW(treeview_frequent), FALSE);
GtkQTreeModel *frequent_model = gtk_q_tree_model_new(
(QAbstractItemModel *)CategorizedBookmarkModel::instance(),
1,
Qt::DisplayRole, G_TYPE_STRING);
gtk_tree_view_set_model(GTK_TREE_VIEW(treeview_frequent), GTK_TREE_MODEL(frequent_model));
/* photo and name/contact method column */
GtkCellArea *area = gtk_cell_area_box_new();
GtkTreeViewColumn *column = gtk_tree_view_column_new_with_area(area);
gtk_tree_view_column_set_title(column, "Name");
/* photo renderer */
GtkCellRenderer *renderer = gtk_cell_renderer_pixbuf_new();
gtk_cell_area_box_pack_start(GTK_CELL_AREA_BOX(area), renderer, FALSE, FALSE, FALSE);
/* get the photo */
gtk_tree_view_column_set_cell_data_func(
column,
renderer,
(GtkTreeCellDataFunc)render_frequen_contact_photo,
NULL,
NULL);
/* name and contact method renderer */
renderer = gtk_cell_renderer_text_new();
g_object_set(G_OBJECT(renderer), "ellipsize", PANGO_ELLIPSIZE_END, NULL);
gtk_cell_area_box_pack_start(GTK_CELL_AREA_BOX(area), renderer, FALSE, FALSE, FALSE);
gtk_tree_view_column_set_cell_data_func(
column,
renderer,
(GtkTreeCellDataFunc)render_frequent_name_and_contact_method,
NULL,
NULL);
gtk_tree_view_append_column(GTK_TREE_VIEW(treeview_frequent), column);
gtk_tree_view_column_set_resizable(column, TRUE);
gtk_tree_view_expand_all(GTK_TREE_VIEW(treeview_frequent));
g_signal_connect(treeview_frequent, "button-press-event", G_CALLBACK(frequent_popup_menu), NULL);
g_signal_connect(treeview_frequent, "row-activated", G_CALLBACK(activate_frequent_item), NULL);
/* contacts */
GtkWidget *label_contacts = gtk_label_new("Contacts");
gtk_box_pack_start(GTK_BOX(vbox_main), label_contacts, FALSE, TRUE, 0);
GtkWidget *treeview_contacts = gtk_tree_view_new();
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview_contacts), FALSE);
gtk_box_pack_start(GTK_BOX(vbox_main), treeview_contacts, FALSE, TRUE, 0);
/* disable default search, we will handle it ourselves via LRC;
* otherwise the search steals input focus on key presses */
gtk_tree_view_set_enable_search(GTK_TREE_VIEW(treeview_contacts), FALSE);
CategorizedContactModel::instance()->setUnreachableHidden(true);
priv->q_contact_model = new ActiveItemProxyModel(CategorizedContactModel::instance());
priv->q_contact_model->setSortRole(Qt::DisplayRole);
priv->q_contact_model->setSortLocaleAware(true);
priv->q_contact_model->setSortCaseSensitivity(Qt::CaseInsensitive);
priv->q_contact_model->sort(0);
GtkQSortFilterTreeModel *contact_model = gtk_q_sort_filter_tree_model_new(
(QSortFilterProxyModel *)priv->q_contact_model,
1,
Qt::DisplayRole, G_TYPE_STRING);
gtk_tree_view_set_model(GTK_TREE_VIEW(treeview_contacts), GTK_TREE_MODEL(contact_model));
/* photo and name/contact method column */
area = gtk_cell_area_box_new();
column = gtk_tree_view_column_new_with_area(area);
gtk_tree_view_column_set_title(column, "Name");
/* photo renderer */
renderer = gtk_cell_renderer_pixbuf_new();
gtk_cell_area_box_pack_start(GTK_CELL_AREA_BOX(area), renderer, FALSE, FALSE, FALSE);
/* get the photo */
gtk_tree_view_column_set_cell_data_func(
column,
renderer,
(GtkTreeCellDataFunc)render_contact_photo,
NULL,
NULL);
/* name and contact method renderer */
renderer = gtk_cell_renderer_text_new();
g_object_set(G_OBJECT(renderer), "ellipsize", PANGO_ELLIPSIZE_END, NULL);
gtk_cell_area_box_pack_start(GTK_CELL_AREA_BOX(area), renderer, FALSE, FALSE, FALSE);
gtk_tree_view_column_set_cell_data_func(
column,
renderer,
(GtkTreeCellDataFunc)render_name_and_contact_method,
NULL,
NULL);
gtk_tree_view_append_column(GTK_TREE_VIEW(treeview_contacts), column);
gtk_tree_view_column_set_resizable(column, TRUE);
gtk_tree_view_expand_all(GTK_TREE_VIEW(treeview_contacts));
g_signal_connect(contact_model, "row-inserted", G_CALLBACK(expand_if_child), treeview_contacts);
g_signal_connect(treeview_contacts, "button-press-event", G_CALLBACK(contacts_popup_menu), treeview_contacts);
g_signal_connect(treeview_contacts, "row-activated", G_CALLBACK(activate_contact_item), NULL);
gtk_widget_show_all(GTK_WIDGET(self));
}
static void
contacts_view_dispose(GObject *object)
{
G_OBJECT_CLASS(contacts_view_parent_class)->dispose(object);
}
static void
contacts_view_finalize(GObject *object)
{
ContactsView *self = CONTACTS_VIEW(object);
ContactsViewPrivate *priv = CONTACTS_VIEW_GET_PRIVATE(self);
delete priv->q_contact_model;
G_OBJECT_CLASS(contacts_view_parent_class)->finalize(object);
}
static void
contacts_view_class_init(ContactsViewClass *klass)
{
G_OBJECT_CLASS(klass)->finalize = contacts_view_finalize;
G_OBJECT_CLASS(klass)->dispose = contacts_view_dispose;
}
GtkWidget *
contacts_view_new()
{
gpointer self = g_object_new(CONTACTS_VIEW_TYPE, NULL);
return (GtkWidget *)self;
}