Mercurial > pidgin
view plugins/icq/gaim_icq.c @ 1971:7fbd745756c0
[gaim-migrate @ 1981]
i thought i fixed this already
committer: Tailor Script <tailor@pidgin.im>
author | Eric Warmenhoven <eric@warmenhoven.org> |
---|---|
date | Thu, 07 Jun 2001 07:36:03 +0000 |
parents | 8ed70631ed15 |
children | 8195d67364a4 |
line wrap: on
line source
#include <gtk/gtk.h> #include <string.h> #include <stdlib.h> #include "icq.h" /* well, we're doing ICQ, right? */ #include "multi.h" /* needed for gaim_connection */ #include "prpl.h" /* needed for prpl */ #include "gaim.h" /* needed for every other damn thing */ #include "proxy.h" #include "pixmaps/gnomeicu-online.xpm" #include "pixmaps/gnomeicu-away.xpm" #include "pixmaps/gnomeicu-dnd.xpm" #include "pixmaps/gnomeicu-na.xpm" #include "pixmaps/gnomeicu-occ.xpm" #include "pixmaps/gnomeicu-ffc.xpm" #define USEROPT_NICK 0 struct icq_data { icq_Link *link; int cur_status; GSList *thru_serv; }; static guint ack_timer = 0; static struct gaim_connection *find_gaim_conn_by_icq_link(icq_Link *link) { GSList *c = connections; struct gaim_connection *gc = NULL; struct icq_data *id; while (c) { gc = (struct gaim_connection *)c->data; if (gc->protocol == PROTO_ICQ) { id = (struct icq_data *)gc->proto_data; if (id->link == link) break; } gc = NULL; c = c->next; } return gc; } static char *icq_name() { return "ICQ"; } static void icq_do_log(icq_Link *link, time_t time, unsigned char level, const char *log) { debug_printf("ICQ debug %d: %s", level, log); } GList *sockets = NULL; struct gaim_sock { int socket; int type; gint inpa; }; static void gaim_icq_handler(gpointer data, gint source, GdkInputCondition cond) { if (cond & GDK_INPUT_READ) icq_HandleReadySocket(source, ICQ_SOCKET_READ); if (cond & GDK_INPUT_WRITE) icq_HandleReadySocket(source, ICQ_SOCKET_WRITE); } static void icq_sock_notify(int socket, int type, int status) { struct gaim_sock *gs; if (status) { GdkInputCondition cond; if (type == ICQ_SOCKET_READ) cond = GDK_INPUT_READ; else cond = GDK_INPUT_WRITE; gs = g_new0(struct gaim_sock, 1); gs->socket = socket; gs->type = type; gs->inpa = gdk_input_add(socket, cond, gaim_icq_handler, NULL); sockets = g_list_append(sockets, gs); debug_printf("Adding socket notifier: %d %d (%d)\n", socket, type, gs->inpa); } else { GList *m = sockets; while (m) { gs = m->data; if ((gs->socket == socket) && (gs->type == type)) break; m = g_list_next(m); } if (m) { gdk_input_remove(gs->inpa); sockets = g_list_remove(sockets, gs); debug_printf("Removing socket notifier: %d %d (%d)\n", socket, type, gs->inpa); g_free(gs); } } } static void icq_online(icq_Link *link) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); struct icq_data *id = (struct icq_data *)gc->proto_data; debug_printf("%s is now online.\n", gc->username); account_online(gc); /*gc->options |= OPT_USR_KEEPALV; this is always-on now */ serv_finish_login(gc); icq_ChangeStatus(id->link, STATUS_ONLINE); } static void icq_logged_off(icq_Link *link) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); struct icq_data *id = (struct icq_data *)gc->proto_data; if (icq_Connect(link, "icq.mirabilis.com", 4000) < 1) { hide_login_progress(gc, "Unable to connect"); signoff(gc); return; } icq_Login(link, STATUS_ONLINE); id->cur_status = STATUS_ONLINE; } static void icq_msg_incoming(icq_Link *link, unsigned long uin, unsigned char hour, unsigned char minute, unsigned char day, unsigned char month, unsigned short year, const char *data) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); char buf[256], *tmp = g_malloc(BUF_LONG); g_snprintf(tmp, BUF_LONG, "%s", data); g_snprintf(buf, sizeof buf, "%lu", uin); serv_got_im(gc, buf, tmp, 0, time((time_t)NULL)); g_free(tmp); } static void icq_user_online(icq_Link *link, unsigned long uin, unsigned long st, unsigned long ip, unsigned short port, unsigned long real_ip, unsigned char tcp_flags) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); guint status; char buf[256]; g_snprintf(buf, sizeof buf, "%lu", uin); status = (st == STATUS_ONLINE) ? UC_NORMAL : UC_UNAVAILABLE | (st << 5); serv_got_update(gc, buf, 1, 0, 0, 0, status, 0); } static void icq_user_offline(icq_Link *link, unsigned long uin) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); char buf[256]; g_snprintf(buf, sizeof buf, "%lu", uin); serv_got_update(gc, buf, 0, 0, 0, 0, 0, 0); } static void icq_user_status(icq_Link *link, unsigned long uin, unsigned long st) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); guint status; char buf[256]; g_snprintf(buf, sizeof buf, "%lu", uin); status = (st == STATUS_ONLINE) ? UC_NORMAL : UC_UNAVAILABLE | (st << 5); serv_got_update(gc, buf, 1, 0, 0, 0, status, 0); } static gint icq_set_timeout_cb() { icq_HandleTimeout(); ack_timer = 0; return FALSE; } static void icq_set_timeout(long interval) { debug_printf("icq_SetTimeout: %ld\n", interval); if (interval > 0 && ack_timer == 0) ack_timer = gtk_timeout_add(interval * 1000, (GtkFunction)icq_set_timeout_cb, NULL); else if (ack_timer > 0) { gtk_timeout_remove(ack_timer); ack_timer = 0; } } static void icq_url_incoming(icq_Link *link, unsigned long uin, unsigned char hour, unsigned char minute, unsigned char day, unsigned char month, unsigned short year, const char *url, const char *descr) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); char *msg = g_malloc(BUF_LONG), buf[256]; g_snprintf(msg, BUF_LONG, "<A HREF=\"%s\">%s</A>", url, descr); g_snprintf(buf, 256, "%lu", uin); serv_got_im(gc, buf, msg, 0, time((time_t)NULL)); g_free(msg); } static void icq_wrong_passwd(icq_Link *link) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); hide_login_progress(gc, "Invalid password."); signoff(gc); } static void icq_invalid_uin(icq_Link *link) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); hide_login_progress(gc, "Invalid UIN."); signoff(gc); } static void icq_info_reply(icq_Link *link, unsigned long uin, const char *nick, const char *first, const char *last, const char *email, char auth) { char buf[16 * 1024]; g_snprintf(buf, sizeof buf, "<B>UIN:</B> %lu<BR>" "<B>Nick:</B> %s<BR>" "<B>Name:</B> %s %s<BR>" "<B>Email:</B> %s\n", uin, nick, first, last, email); g_show_info_text(buf); } static void icq_web_pager(icq_Link *link, unsigned char hour, unsigned char minute, unsigned char day, unsigned char month, unsigned short year, const char *nick, const char *email, const char *msg) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); char *who = g_strdup_printf("ICQ Web Pager: %s (%s)", nick, email); char *what = g_malloc(BUF_LONG); g_snprintf(what, BUF_LONG, "%s", msg); serv_got_im(gc, who, what, 0, time((time_t)NULL)); g_free(who); g_free(what); } static void icq_mail_express(icq_Link *link, unsigned char hour, unsigned char minute, unsigned char day, unsigned char month, unsigned short year, const char *nick, const char *email, const char *msg) { struct gaim_connection *gc = find_gaim_conn_by_icq_link(link); char *who = g_strdup_printf("ICQ Mail Express: %s (%s)", nick, email); char *what = g_malloc(BUF_LONG); g_snprintf(what, BUF_LONG, "%s", msg); serv_got_im(gc, who, what, 0, time((time_t)NULL)); g_free(who); g_free(what); } static void icq_req_not(icq_Link *link, unsigned long id, int type, int arg, void *data) { if (type == ICQ_NOTIFY_FAILED) do_error_dialog("Failure in sending packet", "ICQ error"); return; } static void icq_login(struct aim_user *user) { struct gaim_connection *gc = new_gaim_conn(user); struct icq_data *id = gc->proto_data = g_new0(struct icq_data, 1); icq_Link *link; char ps[9]; icq_LogLevel = ICQ_LOG_MESSAGE; g_snprintf(ps, sizeof(ps), "%s", user->password); link = id->link = icq_ICQLINKNew(atol(user->username), ps, user->proto_opt[USEROPT_NICK][0] ? user->proto_opt[USEROPT_NICK] : "gaim user", TRUE); link->icq_Logged = icq_online; link->icq_Disconnected = icq_logged_off; link->icq_RecvMessage = icq_msg_incoming; link->icq_RecvURL = icq_url_incoming; link->icq_RecvWebPager = icq_web_pager; link->icq_RecvMailExpress = icq_mail_express; link->icq_UserOnline = icq_user_online; link->icq_UserOffline = icq_user_offline; link->icq_UserStatusUpdate = icq_user_status; link->icq_InfoReply = icq_info_reply; link->icq_WrongPassword = icq_wrong_passwd; link->icq_InvalidUIN = icq_invalid_uin; link->icq_Log = icq_do_log; link->icq_RequestNotify = icq_req_not; if (proxytype == PROXY_SOCKS5) icq_SetProxy(link, proxyhost, proxyport, proxyuser[0], proxyuser, proxypass); icq_ContactClear(id->link); if (bud_list_cache_exists(gc)) do_import(NULL, gc); icq_UnsetProxy(link); if (icq_Connect(link, "icq.mirabilis.com", 4000) < 1) { hide_login_progress(gc, "Unable to connect"); signoff(gc); return; } icq_Login(link, STATUS_ONLINE); id->cur_status = STATUS_ONLINE; set_login_progress(gc, 0, "Connecting..."); } static void icq_close(struct gaim_connection *gc) { struct icq_data *id = (struct icq_data *)gc->proto_data; icq_Logout(id->link); icq_Disconnect(id->link); icq_ICQLINKDelete(id->link); g_slist_free(id->thru_serv); g_free(id); } static void icq_send_msg(struct gaim_connection *gc, char *who, char *msg, int away) { if (!away && (strlen(msg) > 0)) { struct icq_data *id = (struct icq_data *)gc->proto_data; GSList *l = id->thru_serv; long w = atol(who); while (l) { if (w == (long)l->data) break; l = l->next; } icq_SendMessage(id->link, w, msg, l ? ICQ_SEND_THRUSERVER : ICQ_SEND_BESTWAY); } } static void icq_keepalive(struct gaim_connection *gc) { struct icq_data *id = (struct icq_data *)gc->proto_data; icq_KeepAlive(id->link); } static void icq_add_buddy(struct gaim_connection *gc, char *who) { struct icq_data *id = (struct icq_data *)gc->proto_data; icq_ContactAdd(id->link, atol(who)); icq_ContactSetVis(id->link, atol(who), TRUE); } static void icq_add_buddies(struct gaim_connection *gc, GList *whos) { struct icq_data *id = (struct icq_data *)gc->proto_data; while (whos) { icq_ContactAdd(id->link, atol(whos->data)); icq_ContactSetVis(id->link, atol(whos->data), TRUE); whos = whos->next; } } static void icq_rem_buddy(struct gaim_connection *gc, char *who) { struct icq_data *id = (struct icq_data *)gc->proto_data; icq_ContactRemove(id->link, atol(who)); } static void icq_set_away(struct gaim_connection *gc, char *state, char *msg) { struct icq_data *id = (struct icq_data *)gc->proto_data; if (gc->away) gc->away = NULL; if (!strcmp(state, "Online")) icq_ChangeStatus(id->link, STATUS_ONLINE); else if (!strcmp(state, "Away")) { icq_ChangeStatus(id->link, STATUS_AWAY); gc->away = ""; } else if (!strcmp(state, "Do Not Disturb")) { icq_ChangeStatus(id->link, STATUS_DND); gc->away = ""; } else if (!strcmp(state, "Not Available")) { icq_ChangeStatus(id->link, STATUS_NA); gc->away = ""; } else if (!strcmp(state, "Occupied")) { icq_ChangeStatus(id->link, STATUS_OCCUPIED); gc->away = ""; } else if (!strcmp(state, "Free For Chat")) { icq_ChangeStatus(id->link, STATUS_FREE_CHAT); gc->away = ""; } else if (!strcmp(state, "Invisible")) { icq_ChangeStatus(id->link, STATUS_INVISIBLE); gc->away = ""; } else if (!strcmp(state, GAIM_AWAY_CUSTOM)) { if (msg) { icq_ChangeStatus(id->link, STATUS_NA); gc->away = ""; } else { icq_ChangeStatus(id->link, STATUS_ONLINE); } } } static char **icq_list_icon(int uc) { guint status; if (uc == UC_NORMAL) return icon_online_xpm; status = uc >> 5; if (status & STATUS_NA) return icon_na_xpm; if (status & STATUS_DND) return icon_dnd_xpm; if (status & STATUS_OCCUPIED) return icon_occ_xpm; if (status & STATUS_AWAY) return icon_away_xpm; if (status & STATUS_FREE_CHAT) return icon_ffc_xpm; if (status & STATUS_INVISIBLE) return NULL; return icon_online_xpm; } static void icq_get_info(struct gaim_connection *gc, char *who) { struct icq_data *id = (struct icq_data *)gc->proto_data; icq_SendInfoReq(id->link, atol(who)); } static void icq_info(GtkObject *obj, char *who) { serv_get_info(gtk_object_get_user_data(obj), who); } static void icq_buddy_menu(GtkWidget *menu, struct gaim_connection *gc, char *who) { GtkWidget *button; button = gtk_menu_item_new_with_label(_("Get Info")); gtk_signal_connect(GTK_OBJECT(button), "activate", GTK_SIGNAL_FUNC(icq_info), who); gtk_object_set_user_data(GTK_OBJECT(button), gc); gtk_menu_append(GTK_MENU(menu), button); gtk_widget_show(button); } static void icq_print_option(GtkEntry *entry, struct aim_user *user) { int entrynum; entrynum = (int) gtk_object_get_user_data(GTK_OBJECT(entry)); if (entrynum == USEROPT_NICK) g_snprintf(user->proto_opt[USEROPT_NICK], sizeof(user->proto_opt[USEROPT_NICK]), "%s", gtk_entry_get_text(entry)); } static void icq_user_opts(GtkWidget *book, struct aim_user *user) { GtkWidget *vbox; GtkWidget *hbox; GtkWidget *label; GtkWidget *entry; vbox = gtk_vbox_new(FALSE, 5); gtk_container_set_border_width(GTK_CONTAINER(vbox), 5); gtk_notebook_append_page(GTK_NOTEBOOK(book), vbox, gtk_label_new("ICQ Options")); gtk_widget_show(vbox); hbox = gtk_hbox_new(FALSE, 5); gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); gtk_widget_show(hbox); label = gtk_label_new("Nick"); gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); gtk_widget_show(label); entry = gtk_entry_new(); gtk_box_pack_end(GTK_BOX(hbox), entry, FALSE, FALSE, 0); gtk_object_set_user_data(GTK_OBJECT(entry), (void *)USEROPT_NICK); gtk_signal_connect(GTK_OBJECT(entry), "changed", GTK_SIGNAL_FUNC(icq_print_option), user); if (user->proto_opt[USEROPT_NICK][0]) gtk_entry_set_text(GTK_ENTRY(entry), user->proto_opt[USEROPT_NICK]); else gtk_entry_set_text(GTK_ENTRY(entry), "gaim user"); gtk_widget_show(entry); hbox = gtk_hbox_new(FALSE, 5); gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); gtk_widget_show(hbox); } static GList *icq_away_states() { GList *m = NULL; m = g_list_append(m, "Online"); m = g_list_append(m, "Away"); m = g_list_append(m, "Do Not Disturb"); m = g_list_append(m, "Not Available"); m = g_list_append(m, "Occupied"); m = g_list_append(m, "Free For Chat"); m = g_list_append(m, "Invisible"); return m; } static void toggle_thru_serv(GtkToggleButton *button, struct conversation *c) { struct gaim_connection *gc = gtk_object_get_user_data(GTK_OBJECT(button)); struct icq_data *id = gc->proto_data; GSList *l = id->thru_serv; long who = atol(c->name); while (l) { if (who == (long)l->data) break; l = l->next; } if (l) id->thru_serv = g_slist_remove(id->thru_serv, (void *)who); else id->thru_serv = g_slist_append(id->thru_serv, (void *)who); } static void icq_insert_convo(struct gaim_connection *gc, struct conversation *c) { GtkWidget *button; button = gtk_check_button_new_with_label("Send message through server"); gtk_box_pack_start(GTK_BOX(c->lbox), button, FALSE, FALSE, 5); gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(toggle_thru_serv), c); gtk_object_set_user_data(GTK_OBJECT(button), gc); gtk_widget_show(button); } static void icq_remove_convo(struct gaim_connection *gc, struct conversation *c) { while (GTK_BOX(c->lbox)->children) gtk_container_remove(GTK_CONTAINER(c->lbox), ((GtkBoxChild *)GTK_BOX(c->lbox)->children->data)->widget); } static struct prpl *my_protocol = NULL; static void icq_init(struct prpl *ret) { ret->protocol = PROTO_ICQ; ret->name = icq_name; ret->list_icon = icq_list_icon; ret->away_states = icq_away_states; ret->buddy_menu = icq_buddy_menu; ret->user_opts = icq_user_opts; ret->insert_convo = icq_insert_convo; ret->remove_convo = icq_remove_convo; ret->login = icq_login; ret->close = icq_close; ret->send_im = icq_send_msg; ret->add_buddy = icq_add_buddy; ret->add_buddies = icq_add_buddies; ret->remove_buddy = icq_rem_buddy; ret->get_info = icq_get_info; ret->set_away = icq_set_away; ret->keepalive = icq_keepalive; my_protocol = ret; } char *gaim_plugin_init(GModule *handle) { icq_SocketNotify = icq_sock_notify; icq_SetTimeout = icq_set_timeout; load_protocol(icq_init, sizeof(struct prpl)); return NULL; } void gaim_plugin_remove() { struct prpl *p = find_prpl(PROTO_ICQ); if (p == my_protocol) unload_protocol(p); } char *name() { return "ICQ"; } char *description() { return "Allows gaim to use the ICQ protocol"; }