| /* |
| * 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 "ringmainwindow.h" |
| |
| #include <gtk/gtk.h> |
| #include "models/gtkqtreemodel.h" |
| #include <callmodel.h> |
| #include <call.h> |
| #include <QtCore/QItemSelectionModel> |
| #include "incomingcallview.h" |
| #include "currentcallview.h" |
| #include <string.h> |
| #include <historymodel.h> |
| #include <contactmethod.h> |
| #include <QtCore/QSortFilterProxyModel> |
| #include "models/gtkqsortfiltertreemodel.h" |
| |
| #define DEFAULT_VIEW_NAME "placeholder" |
| #define VIEW_CONTACTS "contacts" |
| #define VIEW_HISTORY "history" |
| #define VIEW_PRESENCE "presence" |
| |
| struct _RingMainWindow |
| { |
| GtkApplicationWindow parent; |
| }; |
| |
| struct _RingMainWindowClass |
| { |
| GtkApplicationWindowClass parent_class; |
| }; |
| |
| typedef struct _RingMainWindowPrivate RingMainWindowPrivate; |
| |
| struct _RingMainWindowPrivate |
| { |
| GtkWidget *gears; |
| GtkWidget *gears_image; |
| GtkWidget *stack_contacts_history_presence; |
| GtkWidget *radiobutton_contacts; |
| GtkWidget *radiobutton_history; |
| GtkWidget *radiobutton_presence; |
| GtkWidget *treeview_call; |
| GtkWidget *search_entry; |
| GtkWidget *stack_main_view; |
| GtkWidget *button_placecall; |
| }; |
| |
| G_DEFINE_TYPE_WITH_PRIVATE(RingMainWindow, ring_main_window, GTK_TYPE_APPLICATION_WINDOW); |
| |
| #define RING_MAIN_WINDOW_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), RING_MAIN_WINDOW_TYPE, RingMainWindowPrivate)) |
| |
| static QModelIndex |
| get_index_from_selection(GtkTreeSelection *selection) |
| { |
| GtkTreeIter iter; |
| GtkTreeModel *model = NULL; |
| |
| if (gtk_tree_selection_get_selected(selection, &model, &iter)) { |
| return gtk_q_tree_model_get_source_idx(GTK_Q_TREE_MODEL(model), &iter); |
| } else { |
| return QModelIndex(); |
| } |
| } |
| |
| static void |
| update_call_model_selection(GtkTreeSelection *selection, G_GNUC_UNUSED gpointer user_data) |
| { |
| QModelIndex current = get_index_from_selection(selection); |
| if (current.isValid()) |
| CallModel::instance()->selectionModel()->setCurrentIndex(current, QItemSelectionModel::ClearAndSelect); |
| else |
| CallModel::instance()->selectionModel()->clearCurrentIndex(); |
| } |
| |
| static void |
| call_selection_changed(GtkTreeSelection *selection, gpointer win) |
| { |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(RING_MAIN_WINDOW(win)); |
| |
| /* get the current visible stack child */ |
| GtkWidget *old_call_view = gtk_stack_get_visible_child(GTK_STACK(priv->stack_main_view)); |
| |
| QModelIndex idx = get_index_from_selection(selection); |
| if (idx.isValid()) { |
| QVariant state = CallModel::instance()->data(idx, static_cast<int>(Call::Role::LifeCycleState)); |
| GtkWidget *new_call_view = NULL; |
| char* new_call_view_name = NULL; |
| |
| /* FIXME: change when fixed |
| * switch(state.value<Call::LifeCycleState>()) { */ |
| Call::LifeCycleState lifecyclestate = (Call::LifeCycleState)state.toUInt(); |
| switch(lifecyclestate) { |
| case Call::LifeCycleState::INITIALIZATION: |
| case Call::LifeCycleState::FINISHED: |
| new_call_view = incoming_call_view_new(); |
| incoming_call_view_set_call_info(INCOMING_CALL_VIEW(new_call_view), idx); |
| /* use the pointer of the call as a unique name */ |
| new_call_view_name = g_strdup_printf("%p_incoming", (void *)CallModel::instance()->getCall(idx)); |
| break; |
| case Call::LifeCycleState::PROGRESS: |
| new_call_view = current_call_view_new(); |
| current_call_view_set_call_info(CURRENT_CALL_VIEW(new_call_view), idx); |
| /* use the pointer of the call as a unique name */ |
| new_call_view_name = g_strdup_printf("%p_current", (void *)CallModel::instance()->getCall(idx)); |
| break; |
| case Call::LifeCycleState::COUNT__: |
| g_warning("LifeCycleState should never be COUNT"); |
| break; |
| } |
| |
| gtk_stack_add_named(GTK_STACK(priv->stack_main_view), new_call_view, new_call_view_name); |
| gtk_stack_set_visible_child(GTK_STACK(priv->stack_main_view), new_call_view); |
| g_free(new_call_view_name); |
| |
| } else { |
| /* nothing selected in the call model, so show the default screen */ |
| |
| /* TODO: replace stack paceholder view */ |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_main_view), GTK_STACK_TRANSITION_TYPE_SLIDE_LEFT); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_main_view), DEFAULT_VIEW_NAME); |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_main_view), GTK_STACK_TRANSITION_TYPE_SLIDE_RIGHT); |
| |
| } |
| |
| /* check if we changed the visible child */ |
| GtkWidget *current_call_view = gtk_stack_get_visible_child(GTK_STACK(priv->stack_main_view)); |
| if (current_call_view != old_call_view && old_call_view != NULL) { |
| /* if the previous child was a call view, then remove it from |
| * the stack; removing it should destory it since there should not |
| * be any other references to it */ |
| if (IS_INCOMING_CALL_VIEW(old_call_view) || IS_CURRENT_CALL_VIEW(old_call_view)) { |
| gtk_container_remove(GTK_CONTAINER(priv->stack_main_view), old_call_view); |
| } |
| } |
| } |
| |
| static void |
| call_state_changed(Call *call, gpointer win) |
| { |
| g_debug("call state changed"); |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(RING_MAIN_WINDOW(win)); |
| |
| /* check if the call that changed state is the same as the selected call */ |
| QModelIndex idx_selected = CallModel::instance()->selectionModel()->currentIndex(); |
| |
| if( idx_selected.isValid() && call == CallModel::instance()->getCall(idx_selected)) { |
| g_debug("selected call state changed"); |
| /* check if we need to change the view */ |
| GtkWidget *old_call_view = gtk_stack_get_visible_child(GTK_STACK(priv->stack_main_view)); |
| GtkWidget *new_call_view = NULL; |
| QVariant state = CallModel::instance()->data(idx_selected, static_cast<int>(Call::Role::LifeCycleState)); |
| |
| /* check what the current state is vs what is displayed */ |
| |
| /* FIXME: change when fixed |
| * switch(state.value<Call::LifeCycleState>()) { */ |
| Call::LifeCycleState lifecyclestate = (Call::LifeCycleState)state.toUInt(); |
| switch(lifecyclestate) { |
| case Call::LifeCycleState::INITIALIZATION: |
| /* LifeCycleState cannot go backwards, so it should not be possible |
| * that the call is displayed as current (meaning that its in progress) |
| * but have the state 'initialization' */ |
| if (IS_CURRENT_CALL_VIEW(old_call_view)) |
| g_warning("call displayed as current, but is in state of initialization"); |
| break; |
| case Call::LifeCycleState::PROGRESS: |
| if (IS_INCOMING_CALL_VIEW(old_call_view)) { |
| /* change from incoming to current */ |
| new_call_view = current_call_view_new(); |
| current_call_view_set_call_info(CURRENT_CALL_VIEW(new_call_view), idx_selected); |
| /* use the pointer of the call as a unique name */ |
| char* new_call_view_name = NULL; |
| new_call_view_name = g_strdup_printf("%p_current", (void *)CallModel::instance()->getCall(idx_selected)); |
| gtk_stack_add_named(GTK_STACK(priv->stack_main_view), new_call_view, new_call_view_name); |
| g_free(new_call_view_name); |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_main_view), GTK_STACK_TRANSITION_TYPE_SLIDE_UP); |
| gtk_stack_set_visible_child(GTK_STACK(priv->stack_main_view), new_call_view); |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_main_view), GTK_STACK_TRANSITION_TYPE_SLIDE_RIGHT); |
| } |
| break; |
| case Call::LifeCycleState::FINISHED: |
| /* do nothing, either call view is valid for this state */ |
| break; |
| case Call::LifeCycleState::COUNT__: |
| g_warning("LifeCycleState should never be COUNT"); |
| break; |
| } |
| |
| /* check if we changed the visible child */ |
| GtkWidget *current_call_view = gtk_stack_get_visible_child(GTK_STACK(priv->stack_main_view)); |
| if (current_call_view != old_call_view && old_call_view != NULL) { |
| /* if the previous child was a call view, then remove it from |
| * the stack; removing it should destory it since there should not |
| * be any other references to it */ |
| if (IS_INCOMING_CALL_VIEW(old_call_view) || IS_CURRENT_CALL_VIEW(old_call_view)) { |
| gtk_container_remove(GTK_CONTAINER(priv->stack_main_view), old_call_view); |
| } |
| } |
| } |
| } |
| |
| static void |
| search_entry_placecall(G_GNUC_UNUSED GtkWidget *entry, gpointer win) |
| { |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(RING_MAIN_WINDOW(win)); |
| |
| const gchar *number = gtk_entry_get_text(GTK_ENTRY(priv->search_entry)); |
| |
| if (number && strlen(number) > 0) { |
| g_debug("dialing to number: %s", number); |
| Call *call = CallModel::instance()->dialingCall(); |
| call->setDialNumber(number); |
| call->performAction(Call::Action::ACCEPT); |
| |
| /* make this the currently selected call */ |
| QModelIndex idx = CallModel::instance()->getIndex(call); |
| CallModel::instance()->selectionModel()->setCurrentIndex(idx, QItemSelectionModel::ClearAndSelect); |
| } |
| } |
| |
| static void |
| call_history_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); |
| |
| /* 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); |
| |
| QVariant contact_method = idx.data(static_cast<int>(Call::Role::ContactMethod)); |
| /* create new call */ |
| if (contact_method.value<ContactMethod*>()) { |
| Call *call = CallModel::instance()->dialingCall(); |
| call->setDialNumber(contact_method.value<ContactMethod*>()); |
| call->performAction(Call::Action::ACCEPT); |
| |
| /* make this the currently selected call */ |
| QModelIndex call_idx = CallModel::instance()->getIndex(call); |
| CallModel::instance()->selectionModel()->setCurrentIndex(call_idx, QItemSelectionModel::ClearAndSelect); |
| } else |
| g_warning("contact method is empty"); |
| } |
| } |
| |
| static void |
| navbutton_contacts_toggled(GtkToggleButton *navbutton, RingMainWindow *win) |
| { |
| g_return_if_fail(IS_RING_MAIN_WINDOW(win)); |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(win); |
| |
| if (gtk_toggle_button_get_active(navbutton)) { |
| |
| const gchar *visible = gtk_stack_get_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence)); |
| |
| if (visible) { |
| /* contacts is left of both history and presence, so always slide right to show it */ |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_SLIDE_RIGHT); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_CONTACTS); |
| } else { |
| g_warning("should always have a visible child in the nav stack"); |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_NONE); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_CONTACTS); |
| } |
| } |
| } |
| |
| static void |
| navbutton_presence_toggled(GtkToggleButton *navbutton, RingMainWindow *win) |
| { |
| g_return_if_fail(IS_RING_MAIN_WINDOW(win)); |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(win); |
| |
| if (gtk_toggle_button_get_active(navbutton)) { |
| |
| const gchar *visible = gtk_stack_get_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence)); |
| if (visible) { |
| /* presence is right of both history and contacts, so always slide left to show it */ |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_SLIDE_LEFT); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_PRESENCE); |
| } else { |
| g_warning("should always have a visible child in the nav stack"); |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_NONE); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_PRESENCE); |
| } |
| } |
| } |
| |
| static void |
| navbutton_history_toggled(GtkToggleButton *navbutton, RingMainWindow *win) |
| { |
| g_return_if_fail(IS_RING_MAIN_WINDOW(win)); |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(win); |
| |
| if (gtk_toggle_button_get_active(navbutton)) { |
| |
| const gchar *visible = gtk_stack_get_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence)); |
| if (visible) { |
| if (strcmp(visible, VIEW_CONTACTS) == 0) { |
| /* history is right of contacts, so slide left to show it */ |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_SLIDE_LEFT); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_HISTORY); |
| } else if (strcmp(visible, VIEW_PRESENCE) == 0) { |
| /* history is left of presence, so slide right to show it */ |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_SLIDE_RIGHT); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_HISTORY); |
| } |
| } else { |
| g_warning("should always have a visible child in the nav stack"); |
| gtk_stack_set_transition_type(GTK_STACK(priv->stack_contacts_history_presence), GTK_STACK_TRANSITION_TYPE_NONE); |
| gtk_stack_set_visible_child_name(GTK_STACK(priv->stack_contacts_history_presence), VIEW_HISTORY); |
| } |
| } |
| } |
| |
| static void |
| ring_main_window_init(RingMainWindow *win) |
| { |
| RingMainWindowPrivate *priv = RING_MAIN_WINDOW_GET_PRIVATE(win); |
| gtk_widget_init_template(GTK_WIDGET(win)); |
| |
| /* set window icon */ |
| GError *error = NULL; |
| GdkPixbuf* icon = gdk_pixbuf_new_from_resource("/cx/ring/RingGnome/ring-symbol-blue", &error); |
| if (icon == NULL) { |
| g_debug("Could not load icon: %s", error->message); |
| g_error_free(error); |
| } else |
| gtk_window_set_icon(GTK_WINDOW(win), icon); |
| |
| /* set menu icon */ |
| GdkPixbuf* ring_gears = gdk_pixbuf_new_from_resource_at_scale("/cx/ring/RingGnome/ring-logo-blue", |
| -1, 22, TRUE, &error); |
| if (ring_gears == NULL) { |
| g_debug("Could not load icon: %s", error->message); |
| g_error_free(error); |
| } else |
| gtk_image_set_from_pixbuf(GTK_IMAGE(priv->gears_image), ring_gears); |
| |
| /* gears menu */ |
| GtkBuilder *builder = gtk_builder_new_from_resource("/cx/ring/RingGnome/ringgearsmenu.ui"); |
| GMenuModel *menu = G_MENU_MODEL(gtk_builder_get_object(builder, "menu")); |
| gtk_menu_button_set_menu_model(GTK_MENU_BUTTON(priv->gears), menu); |
| g_object_unref(builder); |
| |
| /* call model */ |
| GtkQTreeModel *call_model; |
| GtkCellRenderer *renderer; |
| GtkTreeViewColumn *column; |
| |
| call_model = gtk_q_tree_model_new(CallModel::instance(), 4, |
| Call::Role::Name, G_TYPE_STRING, |
| Call::Role::Number, G_TYPE_STRING, |
| Call::Role::Length, G_TYPE_STRING, |
| Call::Role::State, G_TYPE_STRING); |
| gtk_tree_view_set_model(GTK_TREE_VIEW(priv->treeview_call), GTK_TREE_MODEL(call_model)); |
| |
| renderer = gtk_cell_renderer_text_new(); |
| column = gtk_tree_view_column_new_with_attributes("Name", renderer, "text", 0, NULL); |
| gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE); |
| gtk_tree_view_append_column(GTK_TREE_VIEW(priv->treeview_call), column); |
| |
| renderer = gtk_cell_renderer_text_new(); |
| column = gtk_tree_view_column_new_with_attributes("Duration", renderer, "text", 2, NULL); |
| gtk_tree_view_append_column(GTK_TREE_VIEW(priv->treeview_call), column); |
| |
| /* connect signals to and from UserActionModel to sync selection betwee |
| * the QModel and the GtkTreeView */ |
| QObject::connect( |
| CallModel::instance()->selectionModel(), |
| &QItemSelectionModel::currentChanged, |
| [=](const QModelIndex & current, const QModelIndex & previous) { |
| GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(priv->treeview_call)); |
| |
| /* first unselect the previous */ |
| if (previous.isValid()) { |
| GtkTreeIter old_iter; |
| if (gtk_q_tree_model_source_index_to_iter(call_model, previous, &old_iter)) { |
| gtk_tree_selection_unselect_iter(selection, &old_iter); |
| } else { |
| g_warning("Trying to unselect invalid GtkTreeIter"); |
| } |
| } |
| |
| /* select the current */ |
| if (current.isValid()) { |
| GtkTreeIter new_iter; |
| if (gtk_q_tree_model_source_index_to_iter(call_model, current, &new_iter)) { |
| gtk_tree_selection_select_iter(selection, &new_iter); |
| } else { |
| g_warning("SelectionModel of CallModel changed to invalid QModelIndex?"); |
| } |
| } |
| } |
| ); |
| |
| GtkTreeSelection *call_selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(priv->treeview_call)); |
| g_signal_connect(call_selection, "changed", G_CALLBACK(update_call_model_selection), NULL); |
| |
| /* connect to call state changes to update relevant view(s) */ |
| QObject::connect( |
| CallModel::instance(), |
| &CallModel::callStateChanged, |
| [=](Call* call, G_GNUC_UNUSED Call::State previousState) { |
| call_state_changed(call, win); |
| } |
| ); |
| |
| /* contacts view/model */ |
| GtkWidget *scrolled_window = gtk_scrolled_window_new(NULL, NULL); |
| GtkWidget *treeview_contacts = gtk_tree_view_new(); |
| gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview_contacts), FALSE); |
| gtk_container_add(GTK_CONTAINER(scrolled_window), treeview_contacts); |
| gtk_widget_show_all(scrolled_window); |
| gtk_stack_add_named(GTK_STACK(priv->stack_contacts_history_presence), |
| scrolled_window, |
| VIEW_CONTACTS); |
| |
| /* history view/model */ |
| scrolled_window = gtk_scrolled_window_new(NULL, NULL); |
| GtkWidget *treeview_history = gtk_tree_view_new(); |
| gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview_history), FALSE); |
| gtk_container_add(GTK_CONTAINER(scrolled_window), treeview_history); |
| gtk_widget_show_all(scrolled_window); |
| gtk_stack_add_named(GTK_STACK(priv->stack_contacts_history_presence), |
| scrolled_window, |
| VIEW_HISTORY); |
| /* TODO: make this linked to the client settings so that the last shown view is the same on startup */ |
| gtk_stack_set_visible_child(GTK_STACK(priv->stack_contacts_history_presence), |
| scrolled_window); |
| |
| |
| /* sort the history in descending order by date */ |
| QSortFilterProxyModel *proxyModel = new QSortFilterProxyModel(HistoryModel::instance()); |
| proxyModel->setSourceModel(HistoryModel::instance()); |
| proxyModel->setSortRole(static_cast<int>(Call::Role::Date)); |
| proxyModel->sort(0,Qt::DescendingOrder); |
| |
| GtkQSortFilterTreeModel *history_model = gtk_q_sort_filter_tree_model_new((QSortFilterProxyModel *)proxyModel, 4, |
| Qt::DisplayRole, G_TYPE_STRING, |
| Call::Role::Number, G_TYPE_STRING, |
| Call::Role::FormattedDate, G_TYPE_STRING, |
| Call::Role::Direction, G_TYPE_INT); |
| gtk_tree_view_set_model( GTK_TREE_VIEW(treeview_history), GTK_TREE_MODEL(history_model) ); |
| |
| GtkCellArea *cellarea_name = gtk_cell_area_box_new(); |
| gtk_orientable_set_orientation(GTK_ORIENTABLE(cellarea_name), GTK_ORIENTATION_HORIZONTAL); |
| GtkCellRenderer *cellrenderer_name = gtk_cell_renderer_text_new(); |
| GtkCellRenderer *cellrenderer_number = gtk_cell_renderer_text_new(); |
| |
| gtk_cell_area_box_pack_start(GTK_CELL_AREA_BOX(cellarea_name), |
| cellrenderer_name, |
| FALSE, /* expand */ |
| FALSE, /* align in adjacent rows */ |
| FALSE); /* fixed size in all rows */ |
| |
| gtk_cell_area_box_pack_start(GTK_CELL_AREA_BOX(cellarea_name), |
| cellrenderer_number, |
| FALSE, /* expand */ |
| FALSE, /* align in adjacent rows */ |
| FALSE); /* fixed size in all rows */ |
| |
| column = gtk_tree_view_column_new_with_area(cellarea_name); |
| g_object_set(G_OBJECT(column), "max-width", 250, NULL); |
| |
| gtk_tree_view_column_add_attribute(column, cellrenderer_name, "text", 0); |
| gtk_tree_view_column_add_attribute(column, cellrenderer_number, "text", 1); |
| gtk_tree_view_append_column(GTK_TREE_VIEW(treeview_history), column); |
| |
| renderer = gtk_cell_renderer_text_new (); |
| column = gtk_tree_view_column_new_with_attributes ("Date", renderer, "text", 2, NULL); |
| gtk_tree_view_append_column (GTK_TREE_VIEW (treeview_history), column); |
| |
| gtk_tree_view_expand_all(GTK_TREE_VIEW(treeview_history)); |
| |
| g_signal_connect(treeview_history, "row-activated", G_CALLBACK(call_history_item), NULL); |
| |
| /* presence view/model */ |
| scrolled_window = gtk_scrolled_window_new(NULL, NULL); |
| GtkWidget *treeview_presence = gtk_tree_view_new(); |
| gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview_presence), FALSE); |
| gtk_container_add(GTK_CONTAINER(scrolled_window), treeview_presence); |
| gtk_widget_show_all(scrolled_window); |
| gtk_stack_add_named(GTK_STACK(priv->stack_contacts_history_presence), |
| scrolled_window, |
| VIEW_PRESENCE); |
| |
| /* connect signals to change the contacts/history/presence stack view */ |
| g_signal_connect(priv->radiobutton_contacts, "toggled", G_CALLBACK(navbutton_contacts_toggled), win); |
| g_signal_connect(priv->radiobutton_history, "toggled", G_CALLBACK(navbutton_history_toggled), win); |
| g_signal_connect(priv->radiobutton_presence, "toggled", G_CALLBACK(navbutton_presence_toggled), win); |
| |
| /* TODO: replace stack paceholder view */ |
| GtkWidget *placeholder_view = gtk_tree_view_new(); |
| gtk_widget_show(placeholder_view); |
| gtk_stack_add_named(GTK_STACK(priv->stack_main_view), placeholder_view, DEFAULT_VIEW_NAME); |
| |
| /* connect signals */ |
| g_signal_connect(call_selection, "changed", G_CALLBACK(call_selection_changed), win); |
| g_signal_connect(priv->button_placecall, "clicked", G_CALLBACK(search_entry_placecall), win); |
| g_signal_connect(priv->search_entry, "activate", G_CALLBACK(search_entry_placecall), win); |
| |
| /* style of search entry */ |
| gtk_widget_override_font(priv->search_entry, pango_font_description_from_string("monospace 15")); |
| |
| /* connect to incoming call and focus */ |
| QObject::connect( |
| CallModel::instance(), |
| &CallModel::incomingCall, |
| [=](Call* call) { |
| CallModel::instance()->selectionModel()->setCurrentIndex( |
| CallModel::instance()->getIndex(call), QItemSelectionModel::ClearAndSelect); |
| } |
| ); |
| } |
| |
| static void |
| ring_main_window_finalize(GObject *object) |
| { |
| G_OBJECT_CLASS(ring_main_window_parent_class)->finalize(object); |
| } |
| |
| static void |
| ring_main_window_class_init(RingMainWindowClass *klass) |
| { |
| G_OBJECT_CLASS(klass)->finalize = ring_main_window_finalize; |
| |
| gtk_widget_class_set_template_from_resource(GTK_WIDGET_CLASS (klass), |
| "/cx/ring/RingGnome/ringmainwindow.ui"); |
| |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, treeview_call); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, stack_contacts_history_presence); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, radiobutton_contacts); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, radiobutton_history); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, radiobutton_presence); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, gears); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, gears_image); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, search_entry); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, stack_main_view); |
| gtk_widget_class_bind_template_child_private(GTK_WIDGET_CLASS (klass), RingMainWindow, button_placecall); |
| } |
| |
| GtkWidget * |
| ring_main_window_new (GtkApplication *app) |
| { |
| return (GtkWidget *)g_object_new(RING_MAIN_WINDOW_TYPE, "application", app, NULL); |
| } |