Mercurial > pidgin
view plugins/notify.c @ 3549:face765b6a9d
[gaim-migrate @ 3624]
the last patch fixed most of the problems with typing notification. this finishes
the job. thanks faceprint
committer: Tailor Script <tailor@pidgin.im>
author | Luke Schierer <lschiere@pidgin.im> |
---|---|
date | Wed, 25 Sep 2002 13:19:44 +0000 |
parents | 7e1c6c16dd41 |
children | cd938f18f3f8 |
line wrap: on
line source
/* Rewritten by Etan Reisner <deryni@eden.rutgers.edu> * * Added config dialog * Added control over notification method * Added control over when to release notification * * Thanks to Carles Pina i Estany <carles@pinux.info> * for count of new messages option */ /* if my flash messages patch gets merged in can use cnv->local * to notify on new messages also */ #define GAIM_PLUGINS #include "gaim.h" #include <string.h> #include <ctype.h> #include <stdlib.h> #include <gtk/gtk.h> #include <X11/Xlib.h> #include <X11/Xutil.h> #include <X11/Xatom.h> #include <gdk/gdkx.h> guint choice = 1; #define NOTIFY_FOCUS 0x00000001 #define NOTIFY_TYPE 0x00000002 #define NOTIFY_IN_FOCUS 0x00000004 guint method = 1; #define METHOD_STRING 0x00000001 #define METHOD_QUOTE 0x00000002 #define METHOD_URGENT 0x00000004 #define METHOD_COUNT 0x00000008 void *handle; /* I really don't like this but I was having trouble getting any * other way of removing the signal callbacks to work and not crash gaim */ GtkWidget *really_evil_hack; /* GHashTable *hash = NULL; */ GtkWidget *Dialog = NULL; GtkWidget *Click, *Focus, *Type, *InFocus; GtkWidget *String, *Count, *Quote, *Urgent, *Entry; gchar *title_string = "(*) "; /* predefine some functions, less warnings */ void options(GtkWidget *widget, gpointer data); void un_star(GtkWidget *widget, gpointer data); int un_star_window(GtkWidget *widget, gpointer data); void string_remove(GtkWidget *widget); void count_remove(GtkWidget *widget); void quote_remove(GtkWidget *widget); void urgent_remove(struct conversation *c); int counter (char *buf, int *length); int received_im(struct gaim_connection *gc, char **who, char **what, void *m) { char buf[256]; struct conversation *cnv = find_conversation(*who); GtkWindow *win; char *me = g_strdup(normalize(gc->username)); int revert_to_return; Window focus_return; int c, length; if (!strcmp(me, normalize(*who))) { g_free(me); return 0; } g_free(me); if (cnv == NULL) { if (away_options & OPT_AWAY_QUEUE) return 0; cnv = new_conversation(*who); } win = (GtkWindow *)cnv->window; XGetInputFocus(GDK_WINDOW_XDISPLAY(cnv->window->window), &focus_return, &revert_to_return); if ((choice & NOTIFY_IN_FOCUS) || focus_return != GDK_WINDOW_XWINDOW(cnv->window->window)) { if (method & METHOD_STRING) { strncpy(buf, win->title, sizeof(buf)); if (!strstr(buf, title_string)) { g_snprintf(buf, sizeof(buf), "%s%s", title_string, win->title); gtk_window_set_title(win, buf); } } if (method & METHOD_COUNT) { strncpy(buf, win->title, sizeof(buf)); c = counter(buf, &length); if (!c) { g_snprintf(buf, sizeof(buf), "[1] %s", win->title); } else if (!g_strncasecmp(buf, "[", 1)) { g_snprintf(buf, sizeof(buf), "[%d] %s", c+1, &win->title[3+length]); } gtk_window_set_title(win, buf); } if (method & METHOD_QUOTE) { strncpy(buf, win->title, sizeof(buf)); if (g_strncasecmp(buf, "\"", 1)) { g_snprintf(buf, sizeof(buf), "\"%s\"", win->title); gtk_window_set_title(win, buf); } } if (method & METHOD_URGENT) { /* do it the gdk way for windows compatibility(?) if I can figure it out */ /* Sean says this is a bad thing, and I should try using gtk_property_get first */ /* I'll want to pay attention to note on dev.gnome.org though */ /* gdk_property_change(win->window, WM_HINTS, WM_HINTS, 32, GDK_PROP_MODE_REPLACE, XUrgencyHint, 1); */ XWMHints *hints = XGetWMHints(GDK_WINDOW_XDISPLAY(cnv->window->window), GDK_WINDOW_XWINDOW(cnv->window->window)); hints->flags |= XUrgencyHint; XSetWMHints(GDK_WINDOW_XDISPLAY(cnv->window->window), GDK_WINDOW_XWINDOW(cnv->window->window), hints); } } return 0; } int sent_im(struct gaim_connection *gc, char *who, char **what, void *m) { /* char buf[256]; */ struct conversation *c = find_conversation(who); if (method & METHOD_QUOTE) quote_remove(c->window); if (method & METHOD_COUNT) count_remove(c->window); if (method & METHOD_STRING) string_remove(c->window); if (method & METHOD_URGENT) urgent_remove(c); return 0; } int new_conv(char *who) { struct conversation *c = find_conversation(who); /* g_hash_table_insert(hash, who, GINT_TO_POINTER(choice)); */ if (choice & NOTIFY_FOCUS) { gtk_signal_connect_while_alive(GTK_OBJECT(c->window), "focus-in-event", GTK_SIGNAL_FUNC(un_star), NULL, GTK_OBJECT(really_evil_hack)); gtk_object_set_user_data(GTK_OBJECT(c->window), c); } else { gtk_signal_connect_while_alive(GTK_OBJECT(c->window), "button_press_event", GTK_SIGNAL_FUNC(un_star), NULL, GTK_OBJECT(really_evil_hack)); gtk_object_set_user_data(GTK_OBJECT(c->window), c); gtk_signal_connect_while_alive(GTK_OBJECT(c->text), "button_press_event", GTK_SIGNAL_FUNC(un_star_window), NULL, GTK_OBJECT(really_evil_hack)); gtk_object_set_user_data(GTK_OBJECT(c->text), c); gtk_signal_connect_while_alive(GTK_OBJECT(c->entry), "button_press_event", GTK_SIGNAL_FUNC(un_star_window), NULL, GTK_OBJECT(really_evil_hack)); gtk_object_set_user_data(GTK_OBJECT(c->entry), c); } if (choice & NOTIFY_TYPE) { gtk_signal_connect_while_alive(GTK_OBJECT(c->entry), "key-press-event", GTK_SIGNAL_FUNC(un_star_window), NULL, GTK_OBJECT(really_evil_hack)); gtk_object_set_user_data(GTK_OBJECT(c->entry), (gpointer) c); } return 0; } void un_star(GtkWidget *widget, gpointer data) { struct conversation *c = gtk_object_get_user_data(GTK_OBJECT(widget)); if (method & METHOD_QUOTE) quote_remove(widget); if (method & METHOD_COUNT) count_remove(widget); if (method & METHOD_STRING) string_remove(widget); if (method & METHOD_URGENT) urgent_remove(c); return; } int un_star_window(GtkWidget *widget, gpointer data) { GtkWidget *parent = gtk_widget_get_ancestor(widget, GTK_TYPE_WINDOW); gtk_object_set_user_data(GTK_OBJECT(parent), gtk_object_get_user_data(GTK_OBJECT(widget))); un_star(parent, data); return 0; } /* This function returns the number in [ ]'s or 0 */ int counter (char *buf, int *length) { char temp[256]; int i = 1; *length = 0; /* if (buf[0] != '[') */ /* return (0); */ while (isdigit(buf[i]) && i<sizeof(buf)) { temp[i-1] = buf[i]; (*length)++; i++; } temp[i] = '\0'; if (buf[i] != ']') { *length = 0; return (0); } return (atoi(temp)); } void string_remove(GtkWidget *widget) { char buf[256]; GtkWindow *win = GTK_WINDOW(widget); strncpy(buf, win->title, sizeof(buf)); if (strstr(buf, title_string)) { g_snprintf(buf, sizeof(buf), "%s", &win->title[strlen(title_string)]); gtk_window_set_title(win, buf); } return; } void count_remove(GtkWidget *widget) { char buf[256]; GtkWindow *win = GTK_WINDOW(widget); int length; strncpy(buf, win->title, sizeof(buf)); if (!g_strncasecmp(buf, "[", 1)) { counter(buf, &length); g_snprintf(buf, sizeof(buf), "%s", &win->title[3+length]); gtk_window_set_title(win, buf); } return; } void quote_remove(GtkWidget *widget) { char buf[256]; GtkWindow *win = GTK_WINDOW(widget); strncpy(buf, win->title, sizeof(buf)); if (!g_strncasecmp(buf, "\"", 1)) { g_snprintf(buf, strlen(buf) - 1, "%s", &win->title[1]); gtk_window_set_title(win, buf); } return; } void urgent_remove(struct conversation *c) { GdkWindow *win = c->window->window; XWMHints *hints = XGetWMHints(GDK_WINDOW_XDISPLAY(win), GDK_WINDOW_XWINDOW(win)); hints->flags &= ~XUrgencyHint; XSetWMHints(GDK_WINDOW_XDISPLAY(win), GDK_WINDOW_XWINDOW(win), hints); return; } void save_notify_prefs() { gchar buf[1000]; FILE *fp; snprintf(buf, 1000, "%s/.gaim/.notify", getenv("HOME")); if (!(fp = fopen(buf, "w"))) { do_error_dialog(_("Unable to write to config file"), _("Notify plugin"), GAIM_ERROR); return; } fprintf(fp, "%d=CHOICE\n", choice); fprintf(fp, "%d=METHOD\n", method); fprintf(fp, "%s=STRING\n", title_string); fclose(fp); } void load_notify_prefs() { gchar buf[1000]; gchar **parsed; FILE *fp; g_snprintf(buf, sizeof(buf), "%s/.gaim/.notify", getenv("HOME")); if (!(fp = fopen(buf, "r"))) return; while (fgets(buf, 1000, fp) != NULL) { parsed = g_strsplit(g_strchomp(buf), "=", 2); if (parsed[0] && parsed[1]) { if (!strcmp(parsed[1], "CHOICE")) choice = atoi(parsed[0]); if (!strcmp(parsed[1], "METHOD")) method = atoi(parsed[0]); if (!strcmp(parsed[1], "STRING")) if (title_string != NULL) g_free(title_string); title_string = g_strdup(parsed[0]); } g_strfreev(parsed); } fclose(fp); return; } void options(GtkWidget *widget, gpointer data) { gint option = GPOINTER_TO_INT(data); if (option == 0) choice ^= NOTIFY_FOCUS; else if (option == 1) choice ^= NOTIFY_TYPE; else if (option == 2) { method ^= METHOD_STRING; if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget))) gtk_widget_set_sensitive(Entry, TRUE); else gtk_widget_set_sensitive(Entry, FALSE); } else if (option == 3) method ^= METHOD_QUOTE; else if (option == 4) method ^= METHOD_URGENT; else if (option == 5) choice ^= NOTIFY_IN_FOCUS; else if (option == 6) method ^= METHOD_COUNT; } void setup_buttons() { if (choice & NOTIFY_FOCUS) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Focus), TRUE); else gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Click), TRUE); if (choice & NOTIFY_TYPE) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Type), TRUE); else gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Type), FALSE); if (method & METHOD_STRING) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(String), TRUE); else gtk_widget_set_sensitive(Entry, FALSE); if (method & METHOD_QUOTE) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Quote), TRUE); if (method & METHOD_URGENT) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Urgent), TRUE); if (choice & NOTIFY_IN_FOCUS) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(InFocus), TRUE); if (method & METHOD_COUNT) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(Count), TRUE); return; } void close_dialog(GtkWidget *widget, gpointer data) { gint option = GPOINTER_TO_INT(data); if (option > 0) { title_string = g_strdup(gtk_entry_get_text(GTK_ENTRY(Entry))); save_notify_prefs(); } else if (option < 0) load_notify_prefs(); if (Dialog) gtk_widget_destroy(Dialog); Dialog = NULL; } char *gaim_plugin_init(GModule *hndl) { handle = hndl; really_evil_hack = gtk_label_new(""); /* hash = g_hash_table_new(g_str_hash, g_int_equal); */ load_notify_prefs(); gaim_signal_connect(handle, event_im_recv, received_im, NULL); gaim_signal_connect(handle, event_im_send, sent_im, NULL); gaim_signal_connect(handle, event_new_conversation, new_conv, NULL); return NULL; } void gaim_plugin_remove() { GList *c = conversations; /* guint options; */ gtk_widget_destroy(really_evil_hack); while (c) { struct conversation *cnv = (struct conversation *)c->data; /* if (options = GPOINTER_TO_INT(g_hash_table_lookup(hash, cnv->name))) { */ un_star(cnv->window, NULL); /* if (options & REMOVE_FOCUS) */ /* gtk_signal_disconnect_by_func(GTK_OBJECT(cnv->window), GTK_SIGNAL_FUNC(un_star), NULL); */ /* else { */ /* gtk_signal_disconnect_by_func(GTK_OBJECT(cnv->window), GTK_SIGNAL_FUNC(un_star), NULL); */ /* gtk_signal_disconnect_by_func(GTK_OBJECT(cnv->text), GTK_SIGNAL_FUNC(un_star_window), NULL); */ /* gtk_signal_disconnect_by_func(GTK_OBJECT(cnv->entry), GTK_SIGNAL_FUNC(un_star_window), NULL); */ /* } */ /* if (options & REMOVE_TYPE) */ /* gtk_signal_disconnect_by_func(GTK_OBJECT(cnv->entry), GTK_SIGNAL_FUNC(un_star_window), NULL); */ /* } */ c = c->next; } } char *name() { return "Visual Notification"; } char *description() { return "Puts an asterisk in the title bar of all conversations" " where you have not responded to a message yet."; } void gaim_plugin_config() { GtkWidget *dialog_vbox; GtkWidget *button, *label; GtkWidget *box, *box2, *box3, *box4, *frame; if (Dialog) return; /* main config dialog */ Dialog = gtk_dialog_new(); gtk_window_set_title(GTK_WINDOW(Dialog), "Notify plugin configuration"); gtk_window_set_policy(GTK_WINDOW(Dialog), FALSE, FALSE, TRUE); gtk_signal_connect(GTK_OBJECT(Dialog), "destroy", GTK_SIGNAL_FUNC(close_dialog), GINT_TO_POINTER(-1)); dialog_vbox = GTK_DIALOG(Dialog)->vbox; /* Ok and Cancel buttons */ box = gtk_hbox_new(FALSE, 8); gtk_container_add(GTK_CONTAINER(GTK_DIALOG(Dialog)->action_area), box); button = gtk_button_new_with_label(_("Cancel")); gtk_box_pack_end(GTK_BOX(box), button, FALSE, FALSE, 0); gtk_widget_set_usize(button, 80, -2); gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(close_dialog), GINT_TO_POINTER(0)); button = gtk_button_new_with_label(_("Ok")); gtk_box_pack_start(GTK_BOX(box), button, FALSE, FALSE, 0); gtk_widget_set_usize(button, 80, -2); gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(close_dialog), GINT_TO_POINTER(1)); /* warning label */ label = gtk_label_new(_("Changes in notification removal options take effect only on new conversation windows")); gtk_box_pack_start(GTK_BOX(dialog_vbox), label, FALSE, FALSE, 1); /* main hbox */ box = gtk_hbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(dialog_vbox), box, FALSE, FALSE, 0); box4 = gtk_vbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(box), box4, FALSE, FALSE, 0); /* un-notify choices */ frame = gtk_frame_new(_("Remove notification when:")); gtk_box_pack_start(GTK_BOX(box4), frame, FALSE, FALSE, 0); box2 = gtk_vbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(frame), box2); Focus = gtk_radio_button_new_with_label(NULL, _("Conversation window gains focus.")); gtk_box_pack_start(GTK_BOX(box2), Focus, FALSE, FALSE, 2); Click = gtk_radio_button_new_with_label_from_widget(GTK_RADIO_BUTTON(Focus), _("Conversation window gets clicked.")); gtk_box_pack_start(GTK_BOX(box2), Click, FALSE, FALSE, 2); Type = gtk_check_button_new_with_label(_("Type in conversation window")); gtk_box_pack_start(GTK_BOX(box2), Type, FALSE, FALSE, 2); /* notification method choices */ /* do I need/want any other notification methods? */ frame = gtk_frame_new(_("Notification method:")); gtk_box_pack_start(GTK_BOX(box), frame, FALSE, FALSE, 0); box2 = gtk_vbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(frame), box2); box3 = gtk_hbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(box2), box3, FALSE, FALSE, 0); String = gtk_check_button_new_with_label(_("Insert string into window title:")); gtk_box_pack_start(GTK_BOX(box3), String, FALSE, FALSE, 0); Entry = gtk_entry_new_with_max_length(7); gtk_box_pack_start(GTK_BOX(box3), Entry, FALSE, FALSE, 0); gtk_entry_set_text(GTK_ENTRY(Entry), title_string); Quote = gtk_check_button_new_with_label(_("Quote window title.")); gtk_box_pack_start(GTK_BOX(box2), Quote, FALSE, FALSE, 0); Urgent = gtk_check_button_new_with_label(_("Send URGENT to window manager.")); gtk_box_pack_start(GTK_BOX(box2), Urgent, FALSE, FALSE, 0); label = gtk_label_new(_("Function oddly with tabs:")); gtk_box_pack_start(GTK_BOX(box2), label, FALSE, FALSE, 0); Count = gtk_check_button_new_with_label(_("Insert count of new messages into window title")); gtk_box_pack_start(GTK_BOX(box2), Count, FALSE, FALSE, 0); /* general options */ frame = gtk_frame_new(_("General Options")); gtk_box_pack_start(GTK_BOX(box4), frame, FALSE, FALSE, 0); box = gtk_vbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(frame), box); InFocus = gtk_check_button_new_with_label(_("Notify even when window is in focus")); gtk_box_pack_start(GTK_BOX(box), InFocus, FALSE, FALSE, 0); /* setup buttons, then attach signals */ setup_buttons(); gtk_signal_connect(GTK_OBJECT(Focus), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(0)); gtk_signal_connect(GTK_OBJECT(Type), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(1)); gtk_signal_connect(GTK_OBJECT(String), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(2)); gtk_signal_connect(GTK_OBJECT(Quote), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(3)); gtk_signal_connect(GTK_OBJECT(Urgent), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(4)); gtk_signal_connect(GTK_OBJECT(InFocus), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(5)); gtk_signal_connect(GTK_OBJECT(Count), "toggled", GTK_SIGNAL_FUNC(options), GINT_TO_POINTER(6)); gtk_widget_show_all(Dialog); }