pidgin/pidgin

Use G_DECLARE_DERIVABLE_TYPE for PurpleConversation and additional cleanups

The setters for PurpleConversation are kind of crappy because of the
conversation cache in the purple_conversations api, which we'll address at a
later time.

Testing Done:
Compiled and ran locally, parted a chat and just signed out to verify no new issues.

Reviewed at https://reviews.imfreedom.org/r/613/
/* pidgin
*
* Pidgin is the legal property of its developers, whose names are too numerous
* to list here. Please refer to the COPYRIGHT file distributed with this
* source distribution.
*
* 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 2 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 02111-1301 USA
*/
#include <glib/gi18n-lib.h>
#include <gdk/gdkkeysyms.h>
#include <talkatu.h>
#include <purple.h>
#include "gtkblist.h"
#include "gtknotify.h"
#include "gtkutils.h"
#include "pidgincore.h"
#include "pidginstock.h"
typedef struct
{
GtkWidget *window;
int count;
} PidginUserInfo;
typedef struct
{
PurpleAccount *account;
char *url;
GtkWidget *label;
int count;
gboolean purple_has_handle;
} PidginNotifyMailData;
typedef struct
{
PurpleAccount *account;
GtkListStore *model;
GtkWidget *treeview;
GtkWidget *window;
gpointer user_data;
PurpleNotifySearchResults *results;
} PidginNotifySearchResultsData;
typedef struct
{
PurpleNotifySearchButton *button;
PidginNotifySearchResultsData *data;
} PidginNotifySearchResultsButtonData;
enum
{
PIDGIN_MAIL_ICON,
PIDGIN_MAIL_TEXT,
PIDGIN_MAIL_DATA,
COLUMNS_PIDGIN_MAIL
};
typedef struct
{
/*
* This must be first so PidginNotifyDialog can masquerade as the
* dialog widget.
*/
GtkWidget *dialog;
GtkWidget *treeview;
GtkTreeStore *treemodel;
GtkLabel *label;
GtkWidget *open_button;
GtkWidget *dismiss_button;
GtkWidget *edit_button;
int total_count;
gboolean in_use;
} PidginNotifyDialog;
typedef enum
{
PIDGIN_NOTIFY_MAIL,
PIDGIN_NOTIFY_TYPES
} PidginNotifyType;
static PidginNotifyDialog *mail_dialog = NULL;
static PidginNotifyDialog *pidgin_create_notification_dialog(PidginNotifyType type);
static void *pidgin_notify_emails(PurpleConnection *gc, size_t count, gboolean detailed,
const char **subjects,
const char **froms, const char **tos,
const char **urls);
static void pidgin_close_notify(PurpleNotifyType type, void *ui_handle);
static void
message_response_cb(GtkDialog *dialog, gint id, GtkWidget *widget)
{
purple_notify_close(PURPLE_NOTIFY_MESSAGE, widget);
}
static void
reset_mail_dialog(gpointer unused)
{
g_return_if_fail(mail_dialog != NULL);
if (mail_dialog->in_use)
return;
gtk_widget_destroy(mail_dialog->dialog);
g_free(mail_dialog);
mail_dialog = NULL;
purple_signal_emit(purple_notify_get_handle(), "displaying-emails-clear");
}
gboolean
pidgin_notify_emails_pending()
{
return mail_dialog != NULL
&& !gtk_widget_get_visible(mail_dialog->dialog);
}
void pidgin_notify_emails_present(void *data)
{
if (pidgin_notify_emails_pending()) {
gtk_widget_show_all(mail_dialog->dialog);
mail_dialog->in_use = TRUE;
pidgin_blist_set_headline(NULL, NULL, NULL, NULL, NULL);
mail_dialog->in_use = FALSE;
}
purple_signal_emit(purple_notify_get_handle(), "displaying-emails-clear");
}
static void
email_response_cb(GtkDialog *unused, gint id, PidginNotifyDialog *unused2)
{
PidginNotifyMailData *data = NULL;
GtkTreeModel *model = GTK_TREE_MODEL(mail_dialog->treemodel);
GtkTreeIter iter;
if (id == GTK_RESPONSE_YES)
{
/* A single row activated. Remove that row. */
GtkTreeSelection *selection;
selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(mail_dialog->treeview));
if (gtk_tree_selection_get_selected(selection, NULL, &iter))
{
gtk_tree_model_get(model, &iter, PIDGIN_MAIL_DATA, &data, -1);
purple_notify_uri(NULL, data->url);
gtk_tree_store_remove(mail_dialog->treemodel, &iter);
if (data->purple_has_handle)
purple_notify_close(PURPLE_NOTIFY_EMAILS, data);
else
pidgin_close_notify(PURPLE_NOTIFY_EMAILS, data);
if (gtk_tree_model_get_iter_first(model, &iter))
return;
}
else
return;
}
else
{
/* Remove all the rows */
while (gtk_tree_model_get_iter_first(model, &iter))
{
gtk_tree_model_get(model, &iter, PIDGIN_MAIL_DATA, &data, -1);
if (id == GTK_RESPONSE_ACCEPT)
purple_notify_uri(NULL, data->url);
gtk_tree_store_remove(mail_dialog->treemodel, &iter);
if (data->purple_has_handle)
purple_notify_close(PURPLE_NOTIFY_EMAILS, data);
else
pidgin_close_notify(PURPLE_NOTIFY_EMAILS, data);
}
}
reset_mail_dialog(NULL);
}
static void
email_row_activated_cb(GtkTreeView *tv, GtkTreePath *path,
GtkTreeViewColumn *col, gpointer data)
{
email_response_cb(NULL, GTK_RESPONSE_YES, NULL);
}
static gboolean
formatted_close_cb(GtkWidget *win, GdkEvent *event, void *user_data)
{
purple_notify_close(PURPLE_NOTIFY_FORMATTED, win);
return FALSE;
}
static gboolean
searchresults_close_cb(PidginNotifySearchResultsData *data, GdkEvent *event, gpointer user_data)
{
purple_notify_close(PURPLE_NOTIFY_SEARCHRESULTS, data);
return FALSE;
}
static void
searchresults_callback_wrapper_cb(GtkWidget *widget, PidginNotifySearchResultsButtonData *bd)
{
PidginNotifySearchResultsData *data = bd->data;
GtkTreeSelection *selection;
GtkTreeModel *model;
GtkTreeIter iter;
PurpleNotifySearchButton *button;
GList *row = NULL;
gchar *str;
int i;
g_return_if_fail(data != NULL);
selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(data->treeview));
if (gtk_tree_selection_get_selected(selection, &model, &iter))
{
for (i = 1; i < gtk_tree_model_get_n_columns(GTK_TREE_MODEL(model)); i++) {
gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, i, &str, -1);
row = g_list_append(row, str);
}
}
button = bd->button;
button->callback(purple_account_get_connection(data->account), row, data->user_data);
g_list_free_full(row, g_free);
}
/* copy-paste from gtkrequest.c */
static void
pidgin_widget_decorate_account(GtkWidget *cont, PurpleAccount *account)
{
GtkWidget *image;
GdkPixbuf *pixbuf;
if (!account)
return;
pixbuf = pidgin_create_protocol_icon(account, PIDGIN_PROTOCOL_ICON_SMALL);
image = gtk_image_new_from_pixbuf(pixbuf);
g_object_unref(G_OBJECT(pixbuf));
gtk_widget_set_tooltip_text(image,
purple_account_get_username(account));
if (GTK_IS_BOX(cont)) {
gtk_widget_set_halign(image, GTK_ALIGN_START);
gtk_widget_set_valign(image, GTK_ALIGN_START);
gtk_box_pack_end(GTK_BOX(cont), image, FALSE, TRUE, 0);
}
gtk_widget_show(image);
}
static void *
pidgin_notify_message(PurpleNotifyMessageType type, const char *title,
const char *primary, const char *secondary,
PurpleRequestCommonParameters *cpar)
{
GtkWidget *dialog;
GtkWidget *hbox;
GtkWidget *label;
GtkWidget *img = NULL;
char label_text[2048];
const char *icon_name = NULL;
char *primary_esc, *secondary_esc;
switch (type)
{
case PURPLE_NOTIFY_MSG_ERROR:
icon_name = "dialog-error";
break;
case PURPLE_NOTIFY_MSG_WARNING:
icon_name = "dialog-warning";
break;
case PURPLE_NOTIFY_MSG_INFO:
icon_name = "dialog-information";
break;
default:
icon_name = NULL;
break;
}
if (icon_name != NULL)
{
img = gtk_image_new_from_icon_name(icon_name,
GTK_ICON_SIZE_DIALOG);
gtk_widget_set_halign(img, GTK_ALIGN_START);
gtk_widget_set_valign(img, GTK_ALIGN_START);
}
dialog = gtk_dialog_new_with_buttons(title ? title : PIDGIN_ALERT_TITLE,
NULL, 0, _("Close"),
GTK_RESPONSE_CLOSE, NULL);
gtk_window_set_role(GTK_WINDOW(dialog), "notify_dialog");
g_signal_connect(G_OBJECT(dialog), "response",
G_CALLBACK(message_response_cb), dialog);
gtk_container_set_border_width(GTK_CONTAINER(dialog), 12);
gtk_window_set_resizable(GTK_WINDOW(dialog), FALSE);
gtk_box_set_spacing(GTK_BOX(gtk_dialog_get_content_area(GTK_DIALOG(dialog))),
12);
gtk_container_set_border_width(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dialog))),
6);
hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 12);
gtk_container_add(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dialog))),
hbox);
if (img != NULL)
gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
pidgin_widget_decorate_account(hbox,
purple_request_cpar_get_account(cpar));
primary_esc = g_markup_escape_text(primary, -1);
secondary_esc = (secondary != NULL) ? g_markup_escape_text(secondary, -1) : NULL;
g_snprintf(label_text, sizeof(label_text),
"<span weight=\"bold\" size=\"larger\">%s</span>%s%s",
primary_esc, (secondary ? "\n\n" : ""),
(secondary ? secondary_esc : ""));
g_free(primary_esc);
g_free(secondary_esc);
label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), label_text);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_label_set_selectable(GTK_LABEL(label), TRUE);
gtk_label_set_xalign(GTK_LABEL(label), 0);
gtk_label_set_yalign(GTK_LABEL(label), 0);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
g_object_set_data(G_OBJECT(dialog), "pidgin-parent-from",
purple_request_cpar_get_parent_from(cpar));
pidgin_auto_parent_window(dialog);
gtk_widget_show_all(dialog);
return dialog;
}
static void
selection_changed_cb(GtkTreeSelection *sel, PidginNotifyDialog *dialog)
{
GtkTreeIter iter;
GtkTreeModel *model;
PidginNotifyMailData *data;
gboolean active = TRUE;
if (gtk_tree_selection_get_selected(sel, &model, &iter) == FALSE)
active = FALSE;
else
{
gtk_tree_model_get(model, &iter, PIDGIN_MAIL_DATA, &data, -1);
if (data->url == NULL)
active = FALSE;
}
gtk_widget_set_sensitive(dialog->open_button, active);
}
static void *
pidgin_notify_email(PurpleConnection *gc, const char *subject, const char *from,
const char *to, const char *url)
{
return pidgin_notify_emails(gc, 1, (subject != NULL),
(subject == NULL ? NULL : &subject),
(from == NULL ? NULL : &from),
(to == NULL ? NULL : &to),
(url == NULL ? NULL : &url));
}
static int
mail_window_focus_cb(GtkWidget *widget, GdkEventFocus *focus, gpointer null)
{
gtk_window_set_urgency_hint(GTK_WINDOW(widget), FALSE);
return 0;
}
/* count == 0 means this is a detailed mail notification.
* count > 0 mean non-detailed.
*/
static void *
pidgin_notify_add_mail(GtkTreeStore *treemodel, PurpleAccount *account, char *notification, const char *url, int count, gboolean clear, gboolean *new_data)
{
PidginNotifyMailData *data = NULL;
GtkTreeIter iter;
GdkPixbuf *icon;
gboolean new_n = TRUE;
if (count > 0 || clear) {
/* Allow only one non-detailed email notification for each account */
if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(treemodel), &iter)) {
gboolean advanced;
do {
advanced = FALSE;
gtk_tree_model_get(GTK_TREE_MODEL(treemodel), &iter,
PIDGIN_MAIL_DATA, &data, -1);
if (data && data->account == account) {
if (clear) {
advanced = gtk_tree_store_remove(treemodel, &iter);
mail_dialog->total_count -= data->count;
if (data->purple_has_handle)
purple_notify_close(PURPLE_NOTIFY_EMAILS, data);
else
pidgin_close_notify(PURPLE_NOTIFY_EMAILS, data);
/* We're completely done if we've processed all entries */
if (!advanced)
return NULL;
} else if (data->count > 0) {
new_n = FALSE;
g_free(data->url);
data->url = NULL;
mail_dialog->total_count -= data->count;
break;
}
}
} while (advanced || gtk_tree_model_iter_next(GTK_TREE_MODEL(treemodel), &iter));
}
}
if (clear)
return NULL;
icon = pidgin_create_protocol_icon(account, PIDGIN_PROTOCOL_ICON_MEDIUM);
if (new_n) {
data = g_new0(PidginNotifyMailData, 1);
data->purple_has_handle = TRUE;
gtk_tree_store_append(treemodel, &iter, NULL);
}
if (url != NULL)
data->url = g_strdup(url);
gtk_tree_store_set(treemodel, &iter,
PIDGIN_MAIL_ICON, icon,
PIDGIN_MAIL_TEXT, notification,
PIDGIN_MAIL_DATA, data,
-1);
data->account = account;
/* count == 0 indicates we're adding a single detailed e-mail */
data->count = count > 0 ? count : 1;
if (icon)
g_object_unref(icon);
if (new_data)
*new_data = new_n;
return data;
}
static void *
pidgin_notify_emails(PurpleConnection *gc, size_t count, gboolean detailed,
const char **subjects, const char **froms,
const char **tos, const char **urls)
{
char *notification;
PurpleAccount *account;
PidginNotifyMailData *data = NULL, *data2;
gboolean new_data = FALSE;
GtkTreeSelection *sel;
GtkTreeIter iter;
/* Don't bother updating if there aren't new emails and we don't have any displayed currently */
if (count == 0 && mail_dialog == NULL)
return NULL;
account = purple_connection_get_account(gc);
if (mail_dialog == NULL)
mail_dialog = pidgin_create_notification_dialog(PIDGIN_NOTIFY_MAIL);
mail_dialog->total_count += count;
if (detailed) {
for ( ; count; --count) {
char *to_text = NULL;
char *from_text = NULL;
char *subject_text = NULL;
char *tmp;
gboolean first = TRUE;
if (tos != NULL) {
tmp = g_markup_escape_text(*tos, -1);
to_text = g_strdup_printf("<b>%s</b>: %s\n", _("Account"), tmp);
g_free(tmp);
first = FALSE;
tos++;
}
if (froms != NULL) {
tmp = g_markup_escape_text(*froms, -1);
from_text = g_strdup_printf("%s<b>%s</b>: %s\n", first ? "<br>" : "", _("Sender"), tmp);
g_free(tmp);
first = FALSE;
froms++;
}
if (subjects != NULL) {
tmp = g_markup_escape_text(*subjects, -1);
subject_text = g_strdup_printf("%s<b>%s</b>: %s", first ? "<br>" : "", _("Subject"), tmp);
g_free(tmp);
first = FALSE;
subjects++;
}
#define SAFE(x) ((x) ? (x) : "")
notification = g_strdup_printf("%s%s%s", SAFE(to_text), SAFE(from_text), SAFE(subject_text));
#undef SAFE
g_free(to_text);
g_free(from_text);
g_free(subject_text);
(void)first;
/* If we don't keep track of this, will leak "data" for each of the notifications except the last */
data2 = pidgin_notify_add_mail(mail_dialog->treemodel, account, notification, urls ? *urls : NULL, 0, FALSE, &new_data);
if (data2 && new_data) {
if (data)
data->purple_has_handle = FALSE;
data = data2;
}
g_free(notification);
if (urls != NULL)
urls++;
}
} else {
if (count > 0) {
notification = g_strdup_printf(
ngettext("%s has %d new message.",
"%s has %d new messages.", (int)count),
tos ? *tos : "(unknown)", (int)count);
data2 = pidgin_notify_add_mail(mail_dialog->treemodel, account, notification, urls ? *urls : NULL, count, FALSE, &new_data);
if (data2 && new_data) {
data = data2;
}
g_free(notification);
} else {
/* Clear out all mails for the account */
pidgin_notify_add_mail(mail_dialog->treemodel, account, NULL, NULL, 0, TRUE, NULL);
if (mail_dialog->total_count == 0) {
/*
* There is no API to clear the headline specifically
* This will trigger reset_mail_dialog()
*/
pidgin_blist_set_headline(NULL, NULL, NULL, NULL, NULL);
return NULL;
}
}
}
/* Select first item if nothing selected */
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(mail_dialog->treeview));
if ((gtk_tree_selection_count_selected_rows(sel) < 1)
&& gtk_tree_model_get_iter_first(GTK_TREE_MODEL(mail_dialog->treemodel), &iter)) {
gtk_tree_selection_select_iter(sel, &iter);
}
if (!gtk_widget_get_visible(mail_dialog->dialog)) {
char *label_text = g_strdup_printf(ngettext("<b>%d new email.</b>",
"<b>%d new emails.</b>",
mail_dialog->total_count), mail_dialog->total_count);
mail_dialog->in_use = TRUE; /* So that _set_headline doesn't accidentally
remove the notifications when replacing an
old notification. */
pidgin_blist_set_headline(label_text, "mail-unread",
G_CALLBACK(pidgin_notify_emails_present),
mail_dialog->dialog,
reset_mail_dialog);
mail_dialog->in_use = FALSE;
g_free(label_text);
} else if(!gtk_widget_has_focus(mail_dialog->dialog)) {
gtk_window_set_urgency_hint(GTK_WINDOW(mail_dialog->dialog), TRUE);
}
return data;
}
static gboolean
formatted_input_cb(GtkWidget *win, GdkEventKey *event, gpointer data)
{
if (event->keyval == GDK_KEY_Escape)
{
purple_notify_close(PURPLE_NOTIFY_FORMATTED, win);
return TRUE;
}
return FALSE;
}
static void *
pidgin_notify_formatted(const char *title, const char *primary,
const char *secondary, const char *text)
{
GtkWidget *window;
GtkWidget *vbox;
GtkWidget *label;
GtkWidget *button;
GtkWidget *sw;
GtkWidget *view;
GtkTextBuffer *buffer;
char label_text[2048];
char *linked_text, *primary_esc, *secondary_esc;
window = gtk_dialog_new();
gtk_window_set_title(GTK_WINDOW(window), title);
gtk_window_set_resizable(GTK_WINDOW(window), TRUE);
g_signal_connect(G_OBJECT(window), "delete_event",
G_CALLBACK(formatted_close_cb), NULL);
/* Setup the main vbox */
vbox = gtk_dialog_get_content_area(GTK_DIALOG(window));
/* Setup the descriptive label */
primary_esc = g_markup_escape_text(primary, -1);
secondary_esc = (secondary != NULL) ? g_markup_escape_text(secondary, -1) : NULL;
g_snprintf(label_text, sizeof(label_text),
"<span weight=\"bold\" size=\"larger\">%s</span>%s%s",
primary_esc,
(secondary ? "\n" : ""),
(secondary ? secondary_esc : ""));
g_free(primary_esc);
g_free(secondary_esc);
label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), label_text);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_label_set_selectable(GTK_LABEL(label), TRUE);
gtk_label_set_xalign(GTK_LABEL(label), 0);
gtk_label_set_yalign(GTK_LABEL(label), 0);
gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 0);
gtk_widget_show(label);
/* Add the view */
sw = gtk_scrolled_window_new(NULL, NULL);
gtk_box_pack_start(GTK_BOX(vbox), sw, TRUE, TRUE, 0);
buffer = talkatu_html_buffer_new();
view = talkatu_view_new_with_buffer(buffer);
gtk_container_add(GTK_CONTAINER(sw), view);
gtk_widget_set_name(view, "pidgin_notify_view");
gtk_widget_set_size_request(view, 300, 250);
gtk_widget_show_all(sw);
/* Add the Close button. */
button = gtk_dialog_add_button(GTK_DIALOG(window), _("Close"), GTK_RESPONSE_CLOSE);
gtk_widget_grab_focus(button);
g_signal_connect_swapped(G_OBJECT(button), "clicked",
G_CALLBACK(formatted_close_cb), window);
g_signal_connect(G_OBJECT(window), "key_press_event",
G_CALLBACK(formatted_input_cb), NULL);
/* Make sure URLs are clickable */
linked_text = purple_markup_linkify(text);
talkatu_markup_set_html(TALKATU_BUFFER(buffer), linked_text, -1);
g_free(linked_text);
g_object_set_data(G_OBJECT(window), "view-widget", view);
/* Show the window */
pidgin_auto_parent_window(window);
gtk_widget_show(window);
return window;
}
static void
pidgin_notify_searchresults_new_rows(PurpleConnection *gc, PurpleNotifySearchResults *results,
void *data_)
{
PidginNotifySearchResultsData *data = data_;
GtkListStore *model = data->model;
GtkTreeIter iter;
GdkPixbuf *pixbuf;
GList *row, *column;
guint n;
gtk_list_store_clear(data->model);
pixbuf = pidgin_create_protocol_icon(purple_connection_get_account(gc), PIDGIN_PROTOCOL_ICON_SMALL);
for (row = results->rows; row != NULL; row = row->next) {
gtk_list_store_append(model, &iter);
gtk_list_store_set(model, &iter, 0, pixbuf, -1);
n = 1;
for (column = row->data; column != NULL; column = column->next) {
GValue v;
v.g_type = 0;
g_value_init(&v, G_TYPE_STRING);
g_value_set_string(&v, column->data);
gtk_list_store_set_value(model, &iter, n, &v);
n++;
}
}
if (pixbuf != NULL)
g_object_unref(pixbuf);
}
static void *
pidgin_notify_searchresults(PurpleConnection *gc, const char *title,
const char *primary, const char *secondary,
PurpleNotifySearchResults *results, gpointer user_data)
{
GtkWidget *window;
GtkWidget *treeview;
GtkWidget *close_button;
GType *col_types;
GtkListStore *model;
GtkCellRenderer *renderer;
guint col_num;
GList *columniter;
guint i;
GList *l;
GtkWidget *vbox;
GtkWidget *label;
PidginNotifySearchResultsData *data;
char *label_text;
char *primary_esc, *secondary_esc;
g_return_val_if_fail(gc != NULL, NULL);
g_return_val_if_fail(results != NULL, NULL);
data = g_new0(PidginNotifySearchResultsData, 1);
data->user_data = user_data;
data->results = results;
/* Create the window */
window = gtk_dialog_new();
gtk_window_set_title(GTK_WINDOW(window), title ? title :_("Search Results"));
gtk_container_set_border_width(GTK_CONTAINER(window), 12);
gtk_window_set_resizable(GTK_WINDOW(window), TRUE);
g_signal_connect_swapped(G_OBJECT(window), "delete_event",
G_CALLBACK(searchresults_close_cb), data);
/* Setup the main vbox */
vbox = gtk_dialog_get_content_area(GTK_DIALOG(window));
/* Setup the descriptive label */
primary_esc = (primary != NULL) ? g_markup_escape_text(primary, -1) : NULL;
secondary_esc = (secondary != NULL) ? g_markup_escape_text(secondary, -1) : NULL;
label_text = g_strdup_printf(
"<span weight=\"bold\" size=\"larger\">%s</span>%s%s",
(primary ? primary_esc : ""),
(primary && secondary ? "\n" : ""),
(secondary ? secondary_esc : ""));
g_free(primary_esc);
g_free(secondary_esc);
label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), label_text);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_label_set_xalign(GTK_LABEL(label), 0);
gtk_label_set_yalign(GTK_LABEL(label), 0);
gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 0);
gtk_widget_show(label);
g_free(label_text);
/* +1 is for the automagically created Status column. */
col_num = g_list_length(results->columns) + 1;
/* Setup the list model */
col_types = g_new0(GType, col_num);
/* There always is this first column. */
col_types[0] = GDK_TYPE_PIXBUF;
for (i = 1; i < col_num; i++) {
col_types[i] = G_TYPE_STRING;
}
model = gtk_list_store_newv(col_num, col_types);
g_free(col_types);
/* Setup the treeview */
treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
g_object_unref(G_OBJECT(model));
gtk_widget_set_size_request(treeview, 500, 400);
gtk_tree_selection_set_mode(gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview)),
GTK_SELECTION_SINGLE);
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), TRUE);
gtk_box_pack_start(GTK_BOX(vbox),
pidgin_make_scrollable(treeview, GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS, GTK_SHADOW_IN, -1, -1),
TRUE, TRUE, 0);
gtk_widget_show(treeview);
renderer = gtk_cell_renderer_pixbuf_new();
gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(treeview),
-1, "", renderer, "pixbuf", 0, NULL);
i = 1;
for (columniter = results->columns; columniter != NULL; columniter = columniter->next) {
PurpleNotifySearchColumn *column = columniter->data;
renderer = gtk_cell_renderer_text_new();
gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(treeview), -1,
purple_notify_searchresult_column_get_title(column), renderer, "text", i, NULL);
if (!purple_notify_searchresult_column_is_visible(column))
gtk_tree_view_column_set_visible(gtk_tree_view_get_column(GTK_TREE_VIEW(treeview), i), FALSE);
i++;
}
for (l = results->buttons; l; l = l->next) {
PurpleNotifySearchButton *b = l->data;
GtkWidget *button = NULL;
switch (b->type) {
case PURPLE_NOTIFY_BUTTON_LABELED:
if(b->label) {
button = gtk_dialog_add_button(GTK_DIALOG(window), b->label, GTK_RESPONSE_NONE);
} else {
purple_debug_warning("gtknotify", "Missing button label\n");
}
break;
case PURPLE_NOTIFY_BUTTON_CONTINUE:
button = gtk_dialog_add_button(GTK_DIALOG(window), _("Forward"), GTK_RESPONSE_NONE);
break;
case PURPLE_NOTIFY_BUTTON_ADD:
button = gtk_dialog_add_button(GTK_DIALOG(window), _("Add"), GTK_RESPONSE_NONE);
break;
case PURPLE_NOTIFY_BUTTON_INFO:
button = gtk_dialog_add_button(GTK_DIALOG(window), PIDGIN_STOCK_TOOLBAR_USER_INFO, GTK_RESPONSE_NONE);
break;
case PURPLE_NOTIFY_BUTTON_IM:
button = gtk_dialog_add_button(GTK_DIALOG(window), PIDGIN_STOCK_TOOLBAR_MESSAGE_NEW, GTK_RESPONSE_NONE);
break;
case PURPLE_NOTIFY_BUTTON_JOIN:
button = gtk_dialog_add_button(GTK_DIALOG(window), PIDGIN_STOCK_CHAT, GTK_RESPONSE_NONE);
break;
case PURPLE_NOTIFY_BUTTON_INVITE:
button = gtk_dialog_add_button(GTK_DIALOG(window), PIDGIN_STOCK_INVITE, GTK_RESPONSE_NONE);
break;
default:
purple_debug_warning("gtknotify", "Incorrect button type: %d\n", b->type);
}
if (button != NULL) {
PidginNotifySearchResultsButtonData *bd;
bd = g_new0(PidginNotifySearchResultsButtonData, 1);
bd->button = b;
bd->data = data;
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(searchresults_callback_wrapper_cb), bd);
g_signal_connect_swapped(G_OBJECT(button), "destroy", G_CALLBACK(g_free), bd);
}
}
/* Add the Close button */
close_button = gtk_dialog_add_button(GTK_DIALOG(window), _("Close"), GTK_RESPONSE_CLOSE);
g_signal_connect_swapped(G_OBJECT(close_button), "clicked",
G_CALLBACK(searchresults_close_cb), data);
data->account = purple_connection_get_account(gc);
data->model = model;
data->treeview = treeview;
data->window = window;
/* Insert rows. */
pidgin_notify_searchresults_new_rows(gc, results, data);
/* Show the window */
pidgin_auto_parent_window(window);
gtk_widget_show(window);
return data;
}
/** Xerox'ed from Finch! How the tables have turned!! ;) **/
/** User information. **/
static GHashTable *userinfo;
static char *
userinfo_hash(PurpleAccount *account, const char *who)
{
char key[256];
g_snprintf(key, sizeof(key), "%s - %s",
purple_account_get_username(account),
purple_normalize(account, who));
return g_utf8_strup(key, -1);
}
static void
remove_userinfo(GtkWidget *widget, gpointer key)
{
PidginUserInfo *pinfo = g_hash_table_lookup(userinfo, key);
while (pinfo->count--)
purple_notify_close(PURPLE_NOTIFY_USERINFO, widget);
g_hash_table_remove(userinfo, key);
}
static void *
pidgin_notify_userinfo(PurpleConnection *gc, const char *who,
PurpleNotifyUserInfo *user_info)
{
char *info;
void *ui_handle;
char *key = userinfo_hash(purple_connection_get_account(gc), who);
PidginUserInfo *pinfo = NULL;
if (!userinfo) {
userinfo = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
}
info = purple_notify_user_info_get_text_with_newline(user_info, "<br />");
pinfo = g_hash_table_lookup(userinfo, key);
if (pinfo != NULL) {
GtkWidget *view = g_object_get_data(G_OBJECT(pinfo->window), "view-widget");
GtkTextBuffer *buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(view));
char *linked_text = purple_markup_linkify(info);
talkatu_markup_set_html(TALKATU_BUFFER(buffer), linked_text, -1);
g_free(linked_text);
g_free(key);
ui_handle = pinfo->window;
pinfo->count++;
} else {
char *primary = g_strdup_printf(_("Info for %s"), who);
ui_handle = pidgin_notify_formatted(_("Buddy Information"), primary, NULL, info);
g_signal_handlers_disconnect_by_func(G_OBJECT(ui_handle), G_CALLBACK(formatted_close_cb), NULL);
g_signal_connect(G_OBJECT(ui_handle), "destroy", G_CALLBACK(remove_userinfo), key);
g_free(primary);
pinfo = g_new0(PidginUserInfo, 1);
pinfo->window = ui_handle;
pinfo->count = 1;
g_hash_table_insert(userinfo, key, pinfo);
}
g_free(info);
return ui_handle;
}
static void
pidgin_close_notify(PurpleNotifyType type, void *ui_handle)
{
if (type == PURPLE_NOTIFY_EMAIL || type == PURPLE_NOTIFY_EMAILS)
{
PidginNotifyMailData *data = (PidginNotifyMailData *)ui_handle;
if (data) {
g_free(data->url);
g_free(data);
}
}
else if (type == PURPLE_NOTIFY_SEARCHRESULTS)
{
PidginNotifySearchResultsData *data = (PidginNotifySearchResultsData *)ui_handle;
gtk_widget_destroy(data->window);
purple_notify_searchresults_free(data->results);
g_free(data);
}
else if (ui_handle != NULL)
gtk_widget_destroy(GTK_WIDGET(ui_handle));
}
static void *
pidgin_notify_uri(const char *uri) {
gtk_show_uri_on_window(NULL, uri, GDK_CURRENT_TIME, NULL);
return NULL;
}
static PidginNotifyDialog *
pidgin_create_notification_dialog(PidginNotifyType type)
{
GtkTreeStore *model = NULL;
GtkWidget *dialog = NULL;
GtkWidget *label = NULL;
GtkCellRenderer *rend;
GtkTreeViewColumn *column;
GtkWidget *button = NULL;
GtkWidget *vbox = NULL;
GtkTreeSelection *sel;
PidginNotifyDialog *spec_dialog = NULL;
g_return_val_if_fail(type < PIDGIN_NOTIFY_TYPES, NULL);
if (type == PIDGIN_NOTIFY_MAIL) {
g_return_val_if_fail(mail_dialog == NULL, mail_dialog);
model = gtk_tree_store_new(COLUMNS_PIDGIN_MAIL,
GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_POINTER);
}
dialog = gtk_dialog_new();
/* Vertical box */
vbox = gtk_dialog_get_content_area(GTK_DIALOG(dialog));
/* Setup the dialog */
gtk_container_set_border_width(GTK_CONTAINER(dialog), 6);
gtk_container_set_border_width(GTK_CONTAINER(vbox), 6);
gtk_box_set_spacing(GTK_BOX(vbox), 12);
/* Golden ratio it up! */
gtk_widget_set_size_request(dialog, 550, 400);
spec_dialog = g_new0(PidginNotifyDialog, 1);
spec_dialog->dialog = dialog;
spec_dialog->treemodel = model;
spec_dialog->treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
g_object_unref(G_OBJECT(model));
if (type == PIDGIN_NOTIFY_MAIL) {
gtk_window_set_title(GTK_WINDOW(dialog), _("New Mail"));
gtk_window_set_role(GTK_WINDOW(dialog), "new_mail_detailed");
g_signal_connect(G_OBJECT(dialog), "focus-in-event",
G_CALLBACK(mail_window_focus_cb), NULL);
gtk_dialog_add_button(GTK_DIALOG(dialog),
_("Open All Messages"), GTK_RESPONSE_ACCEPT);
button = gtk_dialog_add_button(GTK_DIALOG(dialog),
PIDGIN_STOCK_OPEN_MAIL, GTK_RESPONSE_YES);
spec_dialog->open_button = button;
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(spec_dialog->treeview), FALSE);
gtk_tree_view_set_search_column(GTK_TREE_VIEW(spec_dialog->treeview), PIDGIN_MAIL_TEXT);
gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(spec_dialog->treeview),
pidgin_tree_view_search_equal_func, NULL, NULL);
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(spec_dialog->treeview));
gtk_tree_selection_set_mode(sel, GTK_SELECTION_BROWSE);
g_signal_connect(G_OBJECT(dialog), "response",
G_CALLBACK(email_response_cb), spec_dialog);
g_signal_connect(G_OBJECT(sel), "changed",
G_CALLBACK(selection_changed_cb), spec_dialog);
g_signal_connect(G_OBJECT(spec_dialog->treeview), "row-activated", G_CALLBACK(email_row_activated_cb), NULL);
column = gtk_tree_view_column_new();
gtk_tree_view_column_set_resizable(column, TRUE);
rend = gtk_cell_renderer_pixbuf_new();
gtk_tree_view_column_pack_start(column, rend, FALSE);
gtk_tree_view_column_set_attributes(column, rend, "pixbuf", PIDGIN_MAIL_ICON, NULL);
rend = gtk_cell_renderer_text_new();
gtk_tree_view_column_pack_start(column, rend, TRUE);
gtk_tree_view_column_set_attributes(column, rend, "markup", PIDGIN_MAIL_TEXT, NULL);
gtk_tree_view_append_column(GTK_TREE_VIEW(spec_dialog->treeview), column);
label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), _("<span weight=\"bold\" size=\"larger\">You have mail!</span>"));
}
gtk_dialog_add_button(GTK_DIALOG(dialog),
_("Close"), GTK_RESPONSE_CLOSE);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_label_set_xalign(GTK_LABEL(label), 0);
gtk_label_set_yalign(GTK_LABEL(label), 0);
gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 0);
gtk_box_pack_start(GTK_BOX(vbox),
pidgin_make_scrollable(spec_dialog->treeview, GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS, GTK_SHADOW_IN, -1, -1),
TRUE, TRUE, 2);
return spec_dialog;
}
static void
signed_off_cb(PurpleConnection *gc, gpointer unused)
{
/* Clear any pending emails for this account */
pidgin_notify_emails(gc, 0, FALSE, NULL, NULL, NULL, NULL);
if (mail_dialog != NULL && mail_dialog->total_count == 0)
reset_mail_dialog(NULL);
}
static void*
pidgin_notify_get_handle(void)
{
static int handle;
return &handle;
}
void pidgin_notify_init(void)
{
void *handle = pidgin_notify_get_handle();
purple_signal_connect(purple_connections_get_handle(), "signed-off",
handle, PURPLE_CALLBACK(signed_off_cb), NULL);
}
void pidgin_notify_uninit(void)
{
purple_signals_disconnect_by_handle(pidgin_notify_get_handle());
}
static PurpleNotifyUiOps ops =
{
pidgin_notify_message,
pidgin_notify_email,
pidgin_notify_emails,
pidgin_notify_formatted,
pidgin_notify_searchresults,
pidgin_notify_searchresults_new_rows,
pidgin_notify_userinfo,
pidgin_notify_uri,
pidgin_close_notify,
NULL,
NULL,
NULL,
NULL
};
PurpleNotifyUiOps *
pidgin_notify_get_ui_ops(void)
{
return &ops;
}