Mercurial > pidgin
view plugins/irc.c @ 1899:a2624692260b
[gaim-migrate @ 1909]
this is part two of three
committer: Tailor Script <tailor@pidgin.im>
author | Eric Warmenhoven <eric@warmenhoven.org> |
---|---|
date | Mon, 28 May 2001 03:53:23 +0000 |
parents | a02584b98823 |
children | b71494004378 |
line wrap: on
line source
/* * gaim - IRC Protocol Plugin * * Copyright (C) 2000-2001, Rob Flynn <rob@tgflinux.com> * Copyright (C) 1998-1999, Mark Spencer <markster@marko.net> * * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ #include "../config.h" #include <netdb.h> #include <gtk/gtk.h> #include <unistd.h> #include <errno.h> #include <netinet/in.h> #include <arpa/inet.h> #include <fcntl.h> #include <string.h> #include <stdlib.h> #include <stdio.h> #include <time.h> #include <sys/socket.h> #include <sys/stat.h> #include <ctype.h> #include "multi.h" #include "prpl.h" #include "gaim.h" #include "proxy.h" #include "pixmaps/free_icon.xpm" #define IRC_BUF_LEN 4096 #define USEROPT_SERV 0 #define USEROPT_PORT 1 static int chat_id = 0; struct irc_channel { int id; gchar *name; }; struct irc_data { int fd; int inpa; /* used for non-block logins */ int timer; int totalblocks; int recblocks; GSList *templist; GList *channels; }; static char *irc_name() { return "IRC"; } static void irc_get_info(struct gaim_connection *gc, char *who); char *name() { return "IRC"; } char *description() { return "Allows gaim to use the IRC protocol"; } static void irc_join_chat(struct gaim_connection *gc, int id, char *name) { struct irc_data *idata = (struct irc_data *)gc->proto_data; gchar *buf = (gchar *) g_malloc(IRC_BUF_LEN + 1); g_snprintf(buf, IRC_BUF_LEN, "JOIN %s\n", name); write(idata->fd, buf, strlen(buf)); write(idata->fd, buf, strlen(buf)); g_free(buf); } static void irc_update_user(struct gaim_connection *gc, char *name, int status) { struct irc_data *idata = (struct irc_data *)gc->proto_data; struct irc_channel *u; GSList *temp = idata->templist; /* Loop through our list */ while (temp) { u = (struct irc_channel *)temp->data; if (g_strcasecmp(u->name, name) == 0) { u->id = status; return; } temp = g_slist_next(temp); } return; } static void irc_request_buddy_update(struct gaim_connection *gc) { struct irc_data *idata = (struct irc_data *)gc->proto_data; GSList *grp = gc->groups; GSList *person; struct group *g; struct buddy *b; struct irc_channel *u; if (idata->templist != NULL) return; idata->recblocks = 0; idata->totalblocks = 1; /* First, let's check to see if we have anyone on our buddylist */ if (!grp) { return; } /* Send the first part of our request */ write(idata->fd, "ISON", 4); /* Step through our list of groups */ while (grp) { g = (struct group *)grp->data; person = g->members; while (person) { b = (struct buddy *)person->data; /* We will store our buddy info here. I know, this is cheap * but hey, its the exact same data structure. Why should we * bother with making another one */ u = g_new0(struct irc_channel, 1); u->id = 0; /* Assume by default that they're offline */ u->name = strdup(b->name); write(idata->fd, " ", 1); write(idata->fd, u->name, strlen(u->name)); idata->templist = g_slist_append(idata->templist, u); person = person->next; } grp = g_slist_next(grp); } write(idata->fd, "\n", 1); } static void irc_send_im(struct gaim_connection *gc, char *who, char *message, int away) { struct irc_data *idata = (struct irc_data *)gc->proto_data; gchar *buf = (gchar *) g_malloc(IRC_BUF_LEN + 1); if (who[0] == '@' || who[0] == '+') { /* If the user trys to msg an op or a voice from the channel, the convo will try * to send it to @nick or +nick... needless to say, this is undesirable. */ who++; } /* Before we actually send this, we should check to see if they're trying * To issue a command and handle it properly. */ if (message[0] == '/') { /* I'll change the implementation of this a little later :-) */ if ((g_strncasecmp(message, "/me ", 4) == 0) && (strlen(message) > 4)) { /* We have /me!! We have /me!! :-) */ gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 4); g_snprintf(buf, IRC_BUF_LEN, "PRIVMSG %s :%cACTION %s%c\n", who, '\001', temp, '\001'); g_free(temp); } else if (!g_strncasecmp(message, "/whois ", 7) && (strlen(message) > 7)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 7); irc_get_info(gc, temp); g_free(temp); return; } } else { g_snprintf(buf, IRC_BUF_LEN, "PRIVMSG %s :%s\n", who, message); } write(idata->fd, buf, strlen(buf)); g_free(buf); } static int find_id_by_name(struct gaim_connection *gc, char *name) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); GList *templist; struct irc_channel *channel; templist = ((struct irc_data *)gc->proto_data)->channels; while (templist) { channel = (struct irc_channel *)templist->data; g_snprintf(temp, IRC_BUF_LEN, "#%s", channel->name); if (g_strcasecmp(temp, name) == 0) { g_free(temp); return channel->id; } templist = templist->next; } g_free(temp); /* Return -1 if we have no ID */ return -1; } static struct irc_channel *find_channel_by_name(struct gaim_connection *gc, char *name) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); GList *templist; struct irc_channel *channel; templist = ((struct irc_data *)gc->proto_data)->channels; while (templist) { channel = (struct irc_channel *)templist->data; g_snprintf(temp, IRC_BUF_LEN, "%s", channel->name); if (g_strcasecmp(temp, name) == 0) { g_free(temp); return channel; } templist = templist->next; } g_free(temp); /* If we found nothing, return nothing :-) */ return NULL; } static struct irc_channel *find_channel_by_id(struct gaim_connection *gc, int id) { struct irc_data *idata = (struct irc_data *)gc->proto_data; struct irc_channel *channel; GList *temp; temp = idata->channels; while (temp) { channel = (struct irc_channel *)temp->data; if (channel->id == id) { /* We've found our man */ return channel; } temp = temp->next; } /* If we didnt find one, return NULL */ return NULL; } static void irc_chat_leave(struct gaim_connection *gc, int id); static void irc_chat_send(struct gaim_connection *gc, int id, char *message) { struct irc_data *idata = (struct irc_data *)gc->proto_data; struct irc_channel *channel = NULL; gchar *buf = (gchar *) g_malloc(IRC_BUF_LEN + 1); char **kick; gboolean is_command = FALSE; /* First lets get our current channel */ channel = find_channel_by_id(gc, id); if (!channel) { /* If for some reason we've lost our channel, let's bolt */ g_free(buf); return; } /* Before we actually send this, we should check to see if they're trying * To issue a command and handle it properly. */ if (message[0] == '/') { if ((g_strncasecmp(message, "/me ", 4) == 0) && (strlen(message) > 4)) { /* We have /me!! We have /me!! :-) */ gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 4); g_snprintf(buf, IRC_BUF_LEN, "PRIVMSG #%s :%cACTION %s%c\n", channel->name, '\001', temp, '\001'); g_free(temp); } else if ((g_strncasecmp(message, "/op ", 4) == 0) && (strlen(message) > 4)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 4); g_snprintf(buf, IRC_BUF_LEN, "MODE #%s +o %s\n", channel->name, temp); g_free(temp); is_command = TRUE; } else if ((g_strncasecmp(message, "/deop ", 6) == 0) && (strlen(message) > 6)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 6); g_snprintf(buf, IRC_BUF_LEN, "MODE #%s -o %s\n", channel->name, temp); g_free(temp); is_command = TRUE; } else if ((g_strncasecmp(message, "/voice ", 7) == 0) && (strlen(message) > 7)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 7); g_snprintf(buf, IRC_BUF_LEN, "MODE #%s +v %s\n", channel->name, temp); g_free(temp); is_command = TRUE; } else if ((g_strncasecmp(message, "/devoice ", 9) == 0) && (strlen(message) > 9)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 6); g_snprintf(buf, IRC_BUF_LEN, "MODE #%s -v %s\n", channel->name, temp); g_free(temp); is_command = TRUE; } else if ((g_strncasecmp(message, "/mode ", 6) == 0) && (strlen(message) > 6)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 6); g_snprintf(buf, IRC_BUF_LEN, "MODE #%s %s\n", channel->name, temp); g_free(temp); is_command = TRUE; } else if (!g_strncasecmp(message, "/whois ", 7) && (strlen(message) > 7)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 7); irc_get_info(gc, temp); g_free(temp); is_command = TRUE; } else if (!g_strncasecmp(message, "/part", 5) && (strlen(message) == 5)) { /* If I'm not mistaken, the chat_leave command was coded under the * pretense that it would only occur when someone closed the window. * For this reason, the /part command will not close the window. Nor * will the window close when the user is /kicked. I'll let you decide * the best way to fix it--I'd imagine it'd just be a little line like * if (convo) close (convo), but I'll let you decide where to put it. */ irc_chat_leave(gc, id); is_command = TRUE; return; } else if (!g_strncasecmp(message, "/join ", 6) && (strlen(message) > 6)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 6); irc_join_chat(gc, 0, temp); g_free(temp); is_command = TRUE; return; } else if (!g_strncasecmp(message, "/raw ", 5) && (strlen(message) > 5)){ gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 5); g_snprintf(buf, IRC_BUF_LEN, "%s\r\n", temp); g_free(temp); is_command = TRUE; } else if (!g_strncasecmp(message, "/quote ", 7) && (strlen(message) >7)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 7); g_snprintf(buf, IRC_BUF_LEN, "%s\r\n", temp); g_free(temp); is_command = TRUE; } else if (!g_strncasecmp(message, "/kick ", 6) && (strlen(message) > 6)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 6); kick = g_strsplit(temp, " ", 2); g_snprintf(buf, IRC_BUF_LEN, "KICK #%s %s :%s\r\n", channel->name, kick[0], kick[1]); g_free(temp); is_command = TRUE; } /* FIXME: I'll go back in and grab this later. -- Rob */ /* I THOUGHT THIS WOULD WORK, BUT I WAS WRONG. WOULD SOMEONE KINDLY FIX IT? else if (!g_strncasecmp(message, "/help", 5)) { gchar *temp = (gchar *) g_malloc(IRC_BUF_LEN + 1); strcpy(temp, message + 5); if (temp == "") { serv_got_chat_in(gc, id, "gAIM", 0, "Available Commands:"); serv_got_chat_in(gc, id, "gAIM", 0, " "); serv_got_chat_in(gc, id, "gAIM", 0, "<b>op voice kick </b>"); serv_got_chat_in(gc, id, "gAIM", 0, "<b>deop devoice whois</b>"); serv_got_chat_in(gc, id, "gAIM", 0, "<b>me raw quote</b>"); serv_got_chat_in(gc, id, "gAIM", 0, "<b>mode</b>"); } else { serv_got_chat_in(gc, id, "gAIM", 0, "Usage: "); if (temp == "op") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/op <nick></b> - Gives operator status to user."); else if (temp == "deop") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/deop <nick></b> - Removes operator status from user."); else if (temp == "me") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/me <action></b> - Sends an action to the channel."); else if (temp == "mode") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/mode {[+|-}|o|p|s|i|t|n|b|v} [<limit][<nick>][<ban mask]</b> - Changes channel and user modes."); else if (temp == "voice") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/voice <nick></b> - Gives voice status to user."); else if (temp == "devoice") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/devoice <nick></b> - Removes voice status from user."); else if (temp == "raw") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/raw <text></b> - Sends raw text to the server."); else if (temp == "kick") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/kick [<comment>]</b> - Kicks a user out of the channel."); else if (temp == "whois") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/whois <nick></b> - Gets information about user."); else if (temp == "quote") serv_got_chat_in(gc, id, "gAIM", 0, "<b>/raw <text></b> - Sends raw text to the server."); else serv_got_chat_in(gc, id, "gAIM", 0, "No such command."); } g_free(temp); is_command = TRUE; } */ } else { g_snprintf(buf, IRC_BUF_LEN, "PRIVMSG #%s :%s\n", channel->name, message); } write(idata->fd, buf, strlen(buf)); /* Since AIM expects us to receive the message we send, we gotta fake it */ if (is_command==FALSE) serv_got_chat_in(gc, id, gc->username, 0, message, time((time_t)NULL)); g_free(buf); } static struct conversation *find_conversation_by_id(struct gaim_connection *gc, int id) { GSList *bc = gc->buddy_chats; struct conversation *b = NULL; while (bc) { b = (struct conversation *)bc->data; if (id == b->id) { break; } bc = bc->next; b = NULL; } if (!b) { return NULL; } return b; } static struct conversation *find_conversation_by_name(struct gaim_connection *gc, char *name) { GSList *bc = gc->buddy_chats; struct conversation *b = NULL; while (bc) { b = (struct conversation *)bc->data; if (g_strcasecmp(name, b->name) == 0) { break; } bc = bc->next; b = NULL; } if (!b) { return NULL; } return b; } static void irc_callback(gpointer data, gint source, GdkInputCondition condition) { struct gaim_connection *gc = data; int i = 0; gchar buf[4096]; gchar **buf2; struct irc_data *idata; idata = (struct irc_data *)gc->proto_data; do { if (read(idata->fd, buf + i, 1) < 0) { hide_login_progress(gc, "Read error"); signoff(gc); return; } } while (buf[i++] != '\n'); buf[--i] = '\0'; g_strchomp(buf); g_print("%s\n", buf); /* Check for errors */ if (((strstr(buf, "ERROR :") && (!strstr(buf, "PRIVMSG ")) && (!strstr(buf, "NOTICE ")) && (strlen(buf) > 7)))) { gchar *u_errormsg; /* Let's get our error message */ u_errormsg = g_strdup(buf + 7); /* We got our error message. Now, let's reaise an * error dialog */ do_error_dialog(u_errormsg, "Gaim: IRC Error"); /* And our necessary garbage collection */ g_free(u_errormsg); return; } /* This should be a whois response. I only care about the first (311) one. I might do * the other's later. They're boring. */ if (((strstr(buf, " 311 ")) && (!strstr(buf, "PRIVMSG")) && (!strstr(buf, "NOTICE")))) { char **res; res = g_strsplit(buf, " ", 7); if (!strcmp(res[1], "311")) { char buf[8192]; g_snprintf(buf, 4096, "<b>Nick:</b> %s<br>" "<b>Host:</b> %s@%s<br>" "<b>Name:</b> %s<br>", res[3], res[4], res[5], res[7]+1); g_show_info_text(buf); } g_strfreev(res); return; } /* Autoresponse to an away message */ if (((strstr(buf, " 301 ")) && (!strstr(buf, "PRIVMSG")) && (!strstr(buf, "NOTICE")))) { char **res; res = g_strsplit(buf, " ", 5); if (!strcmp(res[1], "301")) serv_got_im(gc, res[3], res[4] + 1, 1, time((time_t)NULL)); g_strfreev(res); return; } /* Parse the list of names that we receive when we first sign on to * a channel */ if (((strstr(buf, " 353 ")) && (!strstr(buf, "PRIVMSG")) && (!strstr(buf, "NOTICE")))) { gchar u_host[255]; gchar u_command[32]; gchar u_channel[128]; gchar u_names[IRC_BUF_LEN + 1]; struct conversation *convo = NULL; int j; for (j = 0, i = 0; buf[i] != ' '; j++, i++) { u_host[j] = buf[i]; } u_host[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_command[j] = buf[i]; } u_command[j] = '\0'; i++; for (j = 0; buf[i] != '#'; j++, i++) { } i++; for (j = 0; buf[i] != ':'; j++, i++) { u_channel[j] = buf[i]; } u_channel[j - 1] = '\0'; i++; while ((buf[i] == ' ') || (buf[i] == ':')) { i++; } strcpy(u_names, buf + i); buf2 = g_strsplit(u_names, " ", 0); /* Let's get our conversation window */ convo = find_conversation_by_name(gc, u_channel); if (!convo) { return; } /* Now that we've parsed the hell out of this big * mess, let's try to split up the names properly */ for (i = 0; buf2[i] != NULL; i++) add_chat_buddy(convo, buf2[i]); /* And free our pointers */ g_strfreev(buf2); return; } /* Receive a list of users that are currently online */ if (((strstr(buf, " 303 ")) && (!strstr(buf, "PRIVMSG")) && (!strstr(buf, "NOTICE")))) { gchar u_host[255]; gchar u_command[32]; gchar u_names[IRC_BUF_LEN + 1]; int j; for (j = 0, i = 0; buf[i] != ' '; j++, i++) { u_host[j] = buf[i]; } u_host[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_command[j] = buf[i]; } u_command[j] = '\0'; i++; for (j = 0; buf[i] != ':'; j++, i++) { /* My Nick */ } i++; strcpy(u_names, buf + i); buf2 = g_strsplit(u_names, " ", 0); /* Now that we've parsed the hell out of this big * mess, let's try to split up the names properly */ for (i = 0; buf2[i] != NULL; i++) { /* If we have a name here then our buddy is online. We should * update our temporary gslist accordingly. When we achieve our maximum * list of names then we should force an update */ irc_update_user(gc, buf2[i], 1); } /* Increase our received blocks counter */ idata->recblocks++; /* If we have our total number of blocks */ if (idata->recblocks == idata->totalblocks) { GSList *temp; struct irc_channel *u; /* Let's grab our list of people and bring them all on or off line */ temp = idata->templist; /* Loop */ while (temp) { u = temp->data; /* Tell Gaim to bring the person on or off line */ serv_got_update(gc, u->name, u->id, 0, 0, 0, 0, 0); /* Grab the next entry */ temp = g_slist_next(temp); } /* And now, let's delete all of our entries */ temp = idata->templist; while (temp) { u = temp->data; g_free(u->name); temp = g_slist_remove(temp, u); } /* Reset our list */ idata->totalblocks = 0; idata->recblocks = 0; idata->templist = NULL; return; } /* And free our pointers */ g_strfreev(buf2); return; } if ((strstr(buf, " MODE ")) && (strstr(buf, "!")) && (strstr(buf, "+v") || strstr(buf, "-v") || strstr(buf, "-o") || strstr(buf, "+o")) && (buf[0] == ':') && (!strstr(buf, " NOTICE "))) { gchar u_channel[128]; gchar u_nick[128]; gchar u_mode[5]; char **people; gchar *temp, *temp_new; struct irc_channel *channel; int j; temp = NULL; temp_new = NULL; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; i++; for (j = 0; buf[i] != '#'; j++, i++) { } i++; for (j = 0; buf[i] != ' '; j++, i++) { u_channel[j] = buf[i]; } u_channel[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_mode[j] = buf[i]; } u_mode[j] = '\0'; i++; people = g_strsplit(buf + i, " ", 3); channel = find_channel_by_name(gc, u_channel); if (!channel) { return; } for (j = 0; j < strlen(u_mode) - 1 ; j++) { struct conversation *convo = NULL; convo = find_conversation_by_id(gc, channel->id); temp = (gchar *)g_malloc(strlen(people[j]) + 3); temp_new = (gchar *)g_malloc(strlen(people[j]) + 3); g_snprintf(temp, strlen(people[j]) + 2, "@%s", people[j]); if (u_mode[1] == 'v' && u_mode[0] == '+') { g_snprintf(temp_new, strlen(people[j]) + 2, "+%s", people[j]); } else if (u_mode[1] == 'o' && u_mode[0] == '+') { g_snprintf(temp_new, strlen(people[j]) + 2, "@%s", people[j]); } else if (u_mode[0] == '-') { g_snprintf(temp_new, strlen(people[j]) + 1, "%s", people[j]); } rename_chat_buddy(convo, temp, temp_new); g_snprintf(temp, strlen(people[j]) + 2, "+%s", people[j]); rename_chat_buddy(convo, temp, temp_new); rename_chat_buddy(convo, people[j], temp_new); } if (temp) g_free(temp); if (temp_new) g_free(temp_new); return; } if ((strstr(buf, " KICK ")) && (strstr(buf, "!")) && (buf[0] == ':') && (!strstr(buf, " NOTICE "))) { gchar u_channel[128]; gchar u_nick[128]; gchar u_comment[128]; gchar u_who[128]; int id; gchar *temp; struct irc_channel *channel; int j; temp = NULL; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; i++; for (j = 0; buf[i] != '#'; j++, i++) { } i++; for (j = 0; buf[i] != ' '; j++, i++) { u_channel[j] = buf[i]; } u_channel[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_who[j] = buf[i]; } u_who[j] = '\0'; i++; i++; strcpy(u_comment, buf + i); g_strchomp(u_comment); channel = find_channel_by_name(gc, u_channel); if (!channel) { return; } id = find_id_by_name(gc, u_channel); if (g_strcasecmp(u_nick, gc->username) == 0) { /* It looks like you've been naughty! */ serv_got_chat_left(gc, channel->id); idata->channels = g_list_remove(idata->channels, channel); } else { struct conversation *convo = NULL; /* Find their conversation window */ convo = find_conversation_by_id(gc, channel->id); if (!convo) { /* Some how the window doesn't exist. * Let's get out of here */ return; } /* And remove their name */ /* If the person is an op or voice, this won't work. * so we'll just do a nice hack and remove nick and * @nick and +nick. Truly wasteful. */ temp = (gchar *) g_malloc(strlen(u_who) + 3); g_snprintf(temp, strlen(u_who) + 2, "@%s", u_who); remove_chat_buddy(convo, temp); g_free(temp); temp = (gchar *) g_malloc(strlen(u_who) + 3); g_snprintf(temp, strlen(u_who) + 2, "+%s", u_who); remove_chat_buddy(convo, temp); remove_chat_buddy(convo, u_who); g_free(temp); } /* Go Home! */ return; } if ((strstr(buf, " JOIN ")) && (strstr(buf, "!")) && (buf[0] == ':') && (!strstr(buf, " NOTICE "))) { gchar u_channel[128]; gchar u_nick[128]; struct irc_channel *channel; int j; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; i++; for (j = 0; buf[i] != '#'; j++, i++) { } i++; strcpy(u_channel, buf + i); g_strchomp(u_channel); /* Looks like we're going to join the channel for real * now. Let's create a valid channel structure and add * it to our list. Let's make sure that * we are not already in a channel first */ channel = find_channel_by_name(gc, u_channel); if (!channel) { chat_id++; channel = g_new0(struct irc_channel, 1); channel->id = chat_id; channel->name = strdup(u_channel); idata->channels = g_list_append(idata->channels, channel); serv_got_joined_chat(gc, chat_id, u_channel); } else { struct conversation *convo = NULL; /* Someone else joined. Find their conversation * window */ convo = find_conversation_by_id(gc, channel->id); /* And add their name to it */ add_chat_buddy(convo, u_nick); } return; } if ((strstr(buf, " NICK ")) && (strstr(buf, "!")) && (buf[0] == ':') && (!strstr(buf, " NOTICE "))) { gchar old[128]; gchar new[128]; GList *templist; gchar *temp, *temp_new; struct irc_channel *channel; int j; temp = temp_new = NULL; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { old[j] = buf[i]; } old[j] = '\0'; i++; for (j = 0; buf[i] != ':'; j++, i++) { } i++; strcpy(new, buf + i); g_strchomp(new); templist = ((struct irc_data *)gc->proto_data)->channels; while (templist) { struct conversation *convo = NULL; channel = templist->data; convo = find_conversation_by_id(gc, channel->id); /* If the person is an op or voice, this won't work. * so we'll just do a nice hack and rename nick and * @nick and +nick. Truly wasteful. */ temp = (gchar *)g_malloc(strlen(old) + 5); temp_new = (gchar *)g_malloc(strlen(new) + 5); g_snprintf(temp_new, strlen(new) + 2, "@%s", new); g_snprintf(temp, strlen(old) + 2, "@%s", old); rename_chat_buddy(convo, temp, temp_new); g_snprintf(temp, strlen(old) + 2, "+%s", old); g_snprintf(temp_new, strlen(new) + 2, "+%s", new); rename_chat_buddy(convo, temp, temp_new); rename_chat_buddy(convo, old, new); if (temp) g_free(temp); if (temp_new) g_free(temp_new); templist = templist->next; } return; } if ((strstr(buf, "QUIT ")) && (buf[0] == ':') && (strstr(buf, "!")) && (!strstr(buf, " NOTICE "))) { gchar u_nick[128]; gchar *temp; GList *templist; struct irc_channel *channel; int j; temp = NULL; for (j = 0, i = 1 ; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; templist = ((struct irc_data *)gc->proto_data)->channels; while (templist) { struct conversation *convo = NULL; channel = templist->data; convo = find_conversation_by_id(gc, channel->id); /* If the person is an op or voice, this won't work. * so we'll just do a nice hack and remove nick and * @nick and +nick. Truly wasteful. */ temp = (gchar *) g_malloc(strlen(u_nick) + 2); g_snprintf(temp, strlen(u_nick) + 2, "@%s", u_nick); remove_chat_buddy(convo, temp); g_free(temp); temp = (gchar *) g_malloc(strlen(u_nick) + 2); g_snprintf(temp, strlen(u_nick) + 2, "+%s", u_nick); remove_chat_buddy(convo, temp); remove_chat_buddy(convo, u_nick); templist = templist->next; } g_free(temp); return; } if ((strstr(buf, " PART ")) && (strstr(buf, "!")) && (buf[0] == ':') && (!strstr(buf, " NOTICE "))) { gchar u_channel[128]; gchar u_nick[128]; gchar *temp; struct irc_channel *channel; int j; temp = NULL; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; i++; for (j = 0; buf[i] != '#'; j++, i++) { } i++; for (j = 0; buf[i] != ' '; j++, i++) { if (buf[i] == '\0') { break; } u_channel[j] = buf[i]; } u_channel[j] = '\0'; /* Now, lets check to see if it was US that was leaving. * If so, do the correct thing by closing up all of our * old channel stuff. Otherwise, * we should just print that someone left */ channel = find_channel_by_name(gc, u_channel); if (!channel) { return; } if (g_strcasecmp(u_nick, gc->username) == 0) { /* Looks like we're going to leave the channel for * real now. Let's create a valid channel structure * and add it to our list */ serv_got_chat_left(gc, channel->id); idata->channels = g_list_remove(idata->channels, channel); } else { struct conversation *convo = NULL; /* Find their conversation window */ convo = find_conversation_by_id(gc, channel->id); if (!convo) { /* Some how the window doesn't exist. * Let's get out of here */ return; } /* And remove their name */ /* If the person is an op or voice, this won't work. * so we'll just do a nice hack and remove nick and * @nick and +nick. Truly wasteful. */ temp = (gchar *) g_malloc(strlen(u_nick) + 3); g_snprintf(temp, strlen(u_nick) + 2, "@%s", u_nick); remove_chat_buddy(convo, temp); g_free(temp); temp = (gchar *) g_malloc(strlen(u_nick) + 3); g_snprintf(temp, strlen(u_nick) + 2, "+%s", u_nick); remove_chat_buddy(convo, temp); g_free(temp); remove_chat_buddy(convo, u_nick); } /* Go Home! */ return; } if ((strstr(buf, " NOTICE ")) && (buf[0] == ':')) { gchar u_nick[128]; gchar u_host[255]; gchar u_command[32]; gchar u_channel[128]; gchar u_message[IRC_BUF_LEN]; int j; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_host[j] = buf[i]; } u_host[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_command[j] = buf[i]; } u_command[j] = '\0'; i++; for (j = 0; buf[i] != ':'; j++, i++) { u_channel[j] = buf[i]; } u_channel[j - 1] = '\0'; i++; /* Now that everything is parsed, the rest of this baby must be our message */ strncpy(u_message, buf + i, IRC_BUF_LEN); /* Now, lets check the message to see if there's anything special in it */ if (u_message[0] == '\001') { if ((g_strncasecmp(u_message, "\001PING ", 6) == 0) && (strlen(u_message) > 6)) { /* Someone's triyng to ping us. Let's respond */ gchar u_arg[24]; gchar u_buf[200]; unsigned long tend= time((time_t *)NULL); unsigned long tstart; printf("LA: %s\n", buf); strcpy(u_arg, u_message + 6); u_arg[strlen(u_arg) - 1] = '\0'; tstart = atol(u_arg); g_snprintf(u_buf, sizeof(u_buf), "Ping Reply From %s: [%ld seconds]", u_nick, tend-tstart); do_error_dialog(u_buf, "Gaim IRC - Ping Reply"); return; } } } if ((strstr(buf, " PRIVMSG ")) && (buf[0] == ':')) { gchar u_nick[128]; gchar u_host[255]; gchar u_command[32]; gchar u_channel[128]; gchar u_message[IRC_BUF_LEN]; gboolean is_closing; int j; for (j = 0, i = 1; buf[i] != '!'; j++, i++) { u_nick[j] = buf[i]; } u_nick[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_host[j] = buf[i]; } u_host[j] = '\0'; i++; for (j = 0; buf[i] != ' '; j++, i++) { u_command[j] = buf[i]; } u_command[j] = '\0'; i++; for (j = 0; buf[i] != ':'; j++, i++) { u_channel[j] = buf[i]; } u_channel[j - 1] = '\0'; i++; /* Now that everything is parsed, the rest of this baby must be our message */ strncpy(u_message, buf + i, IRC_BUF_LEN); /* Now, lets check the message to see if there's anything special in it */ if (u_message[0] == '\001') { if (g_strncasecmp(u_message, "\001VERSION", 8) == 0) { /* Looks like we have a version request. Let * us handle it thusly */ g_snprintf(buf, IRC_BUF_LEN, "NOTICE %s :%cVERSION GAIM %s:The Pimpin Penguin AIM Clone:%s%c\n", u_nick, '\001', VERSION, WEBSITE, '\001'); write(idata->fd, buf, strlen(buf)); /* And get the heck out of dodge */ return; } if ((g_strncasecmp(u_message, "\001PING ", 6) == 0) && (strlen(u_message) > 6)) { /* Someone's triyng to ping us. Let's respond */ gchar u_arg[24]; strcpy(u_arg, u_message + 6); u_arg[strlen(u_arg) - 1] = '\0'; g_snprintf(buf, IRC_BUF_LEN, "NOTICE %s :%cPING %s%c\n", u_nick, '\001', u_arg, '\001'); write(idata->fd, buf, strlen(buf)); /* And get the heck out of dodge */ return; } if (g_strncasecmp(u_message, "\001ACTION ", 8) == 0) { /* Looks like we have an action. Let's parse it a little */ strcpy(buf, u_message); strcpy(u_message, "/me "); for (j = 4, i = 8; buf[i] != '\001'; i++, j++) { u_message[j] = buf[i]; } u_message[j] = '\0'; } } /* OK, It is a chat or IM message. Here, let's translate the IRC formatting into * good ol' fashioned gtkimhtml style hypertext markup. */ is_closing = FALSE; while(strchr(u_message, '\002')) { // \002 = ^B gchar *current; gchar *temp, *free_here; temp = g_strdup(strchr(u_message, '\002')); free_here = temp; temp++; current = strchr(u_message, '\002'); *current = '<'; current++; if (is_closing) { *current = '/'; current++; } *current = 'b'; current++; *current = '>'; current++; while (*temp != '\0') { *current = *temp; current++; temp++; } *current = '\0'; g_free(free_here); is_closing = !is_closing; } is_closing = FALSE; while(strchr(u_message, '\037')) { // \037 = ^_ gchar *current; gchar *temp, *free_here; temp = g_strdup(strchr(u_message, '\037')); free_here = temp; temp++; current = strchr(u_message, '\037'); *current = '<'; current++; if (is_closing) { *current = '/'; current++; } *current = 'u'; current++; *current = '>'; current++; while (*temp != '\0') { *current = *temp; current++; temp++; } *current = '\0'; g_free(free_here); is_closing = !is_closing; } while(strchr(u_message, '\003')) { // \003 = ^C /* This is color formatting. IRC uses its own weird little system * that we must translate to HTML. */ /* The format is something like this: * ^C5 or ^C5,3 * The number before the comma is the foreground color, after is the * background color. Either number can be 1 or two digits. */ gchar *current; gchar *temp, *free_here; gchar *font_tag, *body_tag; int fg_color, bg_color; temp = g_strdup(strchr(u_message, '\003')); free_here = temp; temp++; fg_color = bg_color = -1; body_tag = font_tag = ""; /* Parsing the color information: */ do { if (!isdigit(*temp)) break; // This translates to </font> fg_color = (int)(*temp - 48); temp++; if (isdigit(*temp)) { fg_color = (fg_color * 10) + (int)(*temp - 48); temp++; } if (*temp != ',') break; temp++; if (!isdigit(*temp)) break; // This translates to </font> bg_color = (int)(*temp - 48); temp++; if (isdigit(*temp)) { bg_color = (bg_color * 10) + (int)(*temp - 48); temp++; } }while (FALSE); if (fg_color > 15) fg_color = fg_color % 16; if (bg_color > 15) bg_color = bg_color % 16; switch (fg_color) { case -1: font_tag = "</font></body>"; break; case 0: // WHITE font_tag = "<font color=\"#ffffff\">"; /* If no background color is specified, we're going to make it black anyway. * That's probably what the sender anticipated the background color to be. * White on white would be illegible. */ if (bg_color == -1) { body_tag = "<body bgcolor=\"#000000\">"; } break; case 1: // BLACK font_tag = "<font color=\"#000000\">"; break; case 2: // NAVY BLUE font_tag = "<font color=\"#000066\">"; break; case 3: // GREEN font_tag = "<font color=\"#006600\">"; break; case 4: // RED font_tag = "<font color=\"#ff0000\">"; break; case 5: // MAROON font_tag = "<font color=\"#660000\">"; break; case 6: // PURPLE font_tag = "<font color=\"#660066\">"; break; case 7: // DISGUSTING PUKE COLOR font_tag = "<font color=\"#666600\">"; break; case 8: // YELLOW font_tag = "<font color=\"#cccc00\">"; break; case 9: // LIGHT GREEN font_tag = "<font color=\"#33cc33\">"; break; case 10: // TEAL font_tag = "<font color=\"#00acac\">"; break; case 11: // CYAN font_tag = "<font color=\"#00ccac\">"; break; case 12: // BLUE font_tag = "<font color=\"#0000ff\">"; break; case 13: // PINK font_tag = "<font color=\"#cc00cc\">"; break; case 14: // GREY font_tag = "<font color=\"#666666\">"; break; case 15: // SILVER font_tag = "<font color=\"#00ccac\">"; break; } switch (bg_color) { case 0: // WHITE body_tag = "<body bgcolor=\"#ffffff\">"; break; case 1: // BLACK body_tag = "<body bgcolor=\"#000000\">"; break; case 2: // NAVY BLUE body_tag = "<body bgcolor=\"#000066\">"; break; case 3: // GREEN body_tag = "<body bgcolor=\"#006600\">"; break; case 4: // RED body_tag = "<body bgcolor=\"#ff0000\">"; break; case 5: // MAROON body_tag = "<body bgcolor=\"#660000\">"; break; case 6: // PURPLE body_tag = "<body bgcolor=\"#660066\">"; break; case 7: // DISGUSTING PUKE COLOR body_tag = "<body bgcolor=\"#666600\">"; break; case 8: // YELLOW body_tag = "<body bgcolor=\"#cccc00\">"; break; case 9: // LIGHT GREEN body_tag = "<body bgcolor=\"#33cc33\">"; break; case 10: // TEAL body_tag = "<body bgcolor=\"#00acac\">"; break; case 11: // CYAN body_tag = "<body bgcolor=\"#00ccac\">"; break; case 12: // BLUE body_tag = "<body bgcolor=\"#0000ff\">"; break; case 13: // PINK body_tag = "<body bgcolor=\"#cc00cc\">"; break; case 14: // GREY body_tag = "<body bgcolor=\"#666666\">"; break; case 15: // SILVER body_tag = "<body bgcolor=\"#00ccac\">"; break; } current = strchr(u_message, '\003'); while (*body_tag != '\0') { *current = *body_tag; current++; body_tag++; } while (*font_tag != '\0') { *current = *font_tag; current++; font_tag++; } while (*temp != '\0') { *current = *temp; current++; temp++; } *current = '\0'; g_free(free_here); is_closing = !is_closing; } while(strchr(u_message, '\017')) { // \017 = ^O gchar *current; gchar *temp, *free_here; temp = g_strdup(strchr(u_message, '\017')); free_here = temp; temp++; current = strchr(u_message, '\017'); *current = '<'; current++; *current = '/'; current++; *current = 'b'; current++; *current = '>'; current++; *current = '<'; current++; *current = '/'; current++; *current = 'u'; current++; *current = '>'; current++; while (*temp != '\0') { *current = *temp; current++; temp++; } *current = '\0'; g_free(free_here); } /* Let's check to see if we have a channel on our hands */ if (u_channel[0] == '#') { /* Yup. We have a channel */ int id; id = find_id_by_name(gc, u_channel); if (id != -1) { serv_got_chat_in(gc, id, u_nick, 0, u_message, time((time_t)NULL)); } } else { /* Nope. Let's treat it as a private message */ gchar *temp; temp = NULL; temp = (gchar *) g_malloc(strlen(u_nick) + 5); g_snprintf(temp, strlen(u_nick) + 2, "@%s", u_nick); /* If I get a message from SeanEgn, and I already have a window * open for him as @SeanEgn or +SeanEgn, this will keep it in the * same window. Unfortunately, if SeanEgn loses his op status * (a sad thing indeed), the messages will still appear to come from * @SeanEgn, until that convo is closed. */ if (find_conversation(temp)){ serv_got_im(gc, temp, u_message, 0, time((time_t)NULL)); g_free(temp); return; } else { g_snprintf(temp, strlen(u_nick) + 2, "+%s", u_nick); if (find_conversation(temp)) { serv_got_im(gc, temp, u_message, 0, time((time_t)NULL)); g_free(temp); return; } else { g_free(temp); serv_got_im(gc, u_nick, u_message, 0, time((time_t)NULL)); return; } } } return; } /* Let's parse PING requests so that we wont get booted for inactivity */ if (strncmp(buf, "PING :", 6) == 0) { buf2 = g_strsplit(buf, ":", 1); /* Let's build a new response */ g_snprintf(buf, IRC_BUF_LEN, "PONG :%s\n", buf2[1]); write(idata->fd, buf, strlen(buf)); /* And clean up after ourselves */ g_strfreev(buf2); return; } } static void irc_close(struct gaim_connection *gc) { struct irc_data *idata = (struct irc_data *)gc->proto_data; GList *chats = idata->channels; struct irc_channel *cc; gchar *buf = (gchar *) g_malloc(IRC_BUF_LEN); g_snprintf(buf, IRC_BUF_LEN, "QUIT :Download GAIM [%s]\n", WEBSITE); write(idata->fd, buf, strlen(buf)); g_free(buf); if (idata->timer) gtk_timeout_remove(idata->timer); while (chats) { cc = (struct irc_channel *)chats->data; g_free(cc->name); chats = g_list_remove(chats, cc); g_free(cc); } if (gc->inpa) gdk_input_remove(gc->inpa); if (idata->inpa) gdk_input_remove(idata->inpa); close(idata->fd); g_free(gc->proto_data); } static void irc_chat_leave(struct gaim_connection *gc, int id) { struct irc_data *idata = (struct irc_data *)gc->proto_data; struct irc_channel *channel; gchar *buf = (gchar *) g_malloc(IRC_BUF_LEN + 1); channel = find_channel_by_id(gc, id); if (!channel) { return; } g_snprintf(buf, IRC_BUF_LEN, "PART #%s\n", channel->name); write(idata->fd, buf, strlen(buf)); g_free(buf); } static void irc_login_callback(gpointer data, gint source, GdkInputCondition condition) { struct gaim_connection *gc = data; struct irc_data *idata = gc->proto_data; char buf[4096]; if (source == -1) { hide_login_progress(gc, "Write error"); signoff(gc); return; } if (idata->fd == 0) idata->fd = source; g_snprintf(buf, 4096, "NICK %s\n USER %s localhost %s :GAIM (%s)\n", gc->username, g_get_user_name(), gc->user->proto_opt[USEROPT_SERV], WEBSITE); if (write(idata->fd, buf, strlen(buf)) < 0) { hide_login_progress(gc, "Write error"); signoff(gc); return; } idata->inpa = gdk_input_add(idata->fd, GDK_INPUT_READ, irc_callback, gc); idata->inpa = 0; /* Now lets sign ourselves on */ account_online(gc); serv_finish_login(gc); if (bud_list_cache_exists(gc)) do_import(NULL, gc); /* we don't call this now because otherwise some IRC servers might not like us */ idata->timer = gtk_timeout_add(20000, (GtkFunction)irc_request_buddy_update, gc); } static void irc_login(struct aim_user *user) { char buf[4096]; struct gaim_connection *gc = new_gaim_conn(user); struct irc_data *idata = gc->proto_data = g_new0(struct irc_data, 1); g_snprintf(buf, sizeof(buf), "Signon: %s", gc->username); set_login_progress(gc, 2, buf); idata->fd = proxy_connect(user->proto_opt[USEROPT_SERV], user->proto_opt[USEROPT_PORT][0] ? atoi(user->proto_opt[USEROPT_PORT]) : 6667, irc_login_callback, gc); if (!user->gc || (idata->fd < 0)) { hide_login_progress(gc, "Unable to create socket"); signoff(gc); return; } } static void irc_print_option(GtkEntry *entry, struct aim_user *user) { int entrynum; entrynum = (int)gtk_object_get_user_data(GTK_OBJECT(entry)); if (entrynum == USEROPT_SERV) { g_snprintf(user->proto_opt[USEROPT_SERV], sizeof(user->proto_opt[USEROPT_SERV]), "%s", gtk_entry_get_text(entry)); } else if (entrynum == USEROPT_PORT) { g_snprintf(user->proto_opt[USEROPT_PORT], sizeof(user->proto_opt[USEROPT_PORT]), "%s", gtk_entry_get_text(entry)); } } static void irc_user_opts(GtkWidget * book, struct aim_user *user) { /* so here, we create the new notebook page */ 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("IRC 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("Server:"); 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_SERV); gtk_signal_connect(GTK_OBJECT(entry), "changed", GTK_SIGNAL_FUNC(irc_print_option), user); if (user->proto_opt[USEROPT_SERV][0]) { debug_printf("setting text %s\n", user->proto_opt[USEROPT_SERV]); gtk_entry_set_text(GTK_ENTRY(entry), user->proto_opt[USEROPT_SERV]); } gtk_widget_show(entry); hbox = gtk_hbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); gtk_widget_show(hbox); label = gtk_label_new("Port:"); 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_PORT); gtk_signal_connect(GTK_OBJECT(entry), "changed", GTK_SIGNAL_FUNC(irc_print_option), user); if (user->proto_opt[USEROPT_PORT][0]) { debug_printf("setting text %s\n", user->proto_opt[USEROPT_PORT]); gtk_entry_set_text(GTK_ENTRY(entry), user->proto_opt[USEROPT_PORT]); } else gtk_entry_set_text(GTK_ENTRY(entry), "6667"); gtk_widget_show(entry); } static char **irc_list_icon(int uc) { return free_icon_xpm; } /* Send out a ping request to the specified user */ static void irc_send_ping(GtkObject * w, char *who) { struct gaim_connection *gc = (struct gaim_connection *)gtk_object_get_user_data(w); struct irc_data *idata = (struct irc_data *)gc->proto_data; char buf[BUF_LEN]; g_snprintf(buf, BUF_LEN, "PRIVMSG %s :%cPING %ld%c\n", who, '\001', time((time_t *)NULL), '\001'); write(idata->fd, buf, strlen(buf)); } /* Do a whois check on someone :-) */ static void irc_get_info(struct gaim_connection *gc, char *who) { struct irc_data *idata = (struct irc_data *)gc->proto_data; char buf[BUF_LEN]; if (((who[0] == '@') || (who[0] == '+')) && (strlen(who)>1)) g_snprintf(buf, BUF_LEN, "WHOIS %s\n", who+1); else g_snprintf(buf, BUF_LEN, "WHOIS %s\n", who); write(idata->fd, buf, strlen(buf)); } static void irc_send_whois(GtkObject * w, char *who) { struct gaim_connection *gc = (struct gaim_connection *)gtk_object_get_user_data(w); irc_get_info(gc, who); } static void irc_buddy_menu(GtkWidget * menu, struct gaim_connection *gc, char *who) { GtkWidget *button; button = gtk_menu_item_new_with_label("Ping"); gtk_signal_connect(GTK_OBJECT(button), "activate", GTK_SIGNAL_FUNC(irc_send_ping), who); gtk_object_set_user_data(GTK_OBJECT(button), gc); gtk_menu_append(GTK_MENU(menu), button); gtk_widget_show(button); button = gtk_menu_item_new_with_label("Whois"); gtk_signal_connect(GTK_OBJECT(button), "activate", GTK_SIGNAL_FUNC(irc_send_whois), who); gtk_object_set_user_data(GTK_OBJECT(button), gc); gtk_menu_append(GTK_MENU(menu), button); gtk_widget_show(button); } static void irc_set_away(struct gaim_connection *gc, char *state, char *msg) { struct irc_data *idata = (struct irc_data *)gc->proto_data; char buf[BUF_LEN]; if (msg) g_snprintf(buf, BUF_LEN, "AWAY :%s\n", msg); else g_snprintf(buf, BUF_LEN, "AWAY\n"); write(idata->fd, buf, strlen(buf)); } static void irc_fake_buddy(struct gaim_connection *gc, char *who) { /* Heh, there is no buddy list. We fake it. * I just need this here so the add and remove buttons will * show up */ } static struct prpl *my_protocol = NULL; static void irc_init(struct prpl *ret) { ret->protocol = PROTO_IRC; ret->name = irc_name; ret->list_icon = irc_list_icon; ret->buddy_menu = irc_buddy_menu; ret->user_opts = irc_user_opts; ret->login = irc_login; ret->close = irc_close; ret->send_im = irc_send_im; ret->join_chat = irc_join_chat; ret->chat_leave = irc_chat_leave; ret->chat_send = irc_chat_send; ret->get_info = irc_get_info; ret->set_away = irc_set_away; ret->add_buddy = irc_fake_buddy; ret->remove_buddy = irc_fake_buddy; my_protocol = ret; } char *gaim_plugin_init(GModule * handle) { load_protocol(irc_init, sizeof(struct prpl)); return NULL; } void gaim_plugin_remove() { struct prpl *p = find_prpl(PROTO_IRC); if (p == my_protocol) unload_protocol(p); }