Mercurial > pidgin
view src/protocols/qq/qq.c @ 14015:9516a796ed5f
[gaim-migrate @ 16607]
Added a debugging tool for firing custom packets at the QQ server.
committer: Tailor Script <tailor@pidgin.im>
author | Mark Huetsch <markhuetsch> |
---|---|
date | Tue, 01 Aug 2006 17:39:47 +0000 |
parents | 9c5790820ac6 |
children | 39d6d4128599 |
line wrap: on
line source
/** * @file qq.c The QQ2003C protocol plugin * * gaim * * Copyright (C) 2004 Puzzlebird * * 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 "internal.h" #ifdef _WIN32 #define random rand #endif #include "debug.h" #include "notify.h" #include "prefs.h" #include "request.h" #include "accountopt.h" #include "prpl.h" #include "server.h" #include "util.h" /* GaimMenuAction, gaim_menu_action_new */ #include "utils.h" #include "buddy_info.h" #include "buddy_opt.h" #include "buddy_status.h" #include "char_conv.h" #include "group_find.h" /* qq_group_find_member_by_channel_and_nickname */ #include "group_im.h" /* qq_send_packet_group_im */ #include "group_info.h" /* qq_send_cmd_group_get_group_info */ #include "group_join.h" /* qq_group_join */ #include "group_opt.h" /* qq_group_manage_members */ #include "group.h" /* chat_info, etc */ #include "header_info.h" /* qq_get_cmd_desc */ #include "im.h" #include "keep_alive.h" #include "ip_location.h" /* qq_ip_get_location */ #include "login_logout.h" #include "packet_parse.h" /* MAX_PACKET_SIZE */ #include "qq_proxy.h" /* qq_connect, qq_disconnect */ #include "send_core.h" #include "qq.h" #include "send_file.h" #include "version.h" #define OPENQ_AUTHOR "Puzzlebird" #define OPENQ_WEBSITE "http://openq.sourceforge.net" #define QQ_TCP_QUERY_PORT "8000" #define QQ_UDP_PORT "8000" const gchar *udp_server_list[] = { "sz.tencent.com", // 61.144.238.145 "sz2.tencent.com", // 61.144.238.146 "sz3.tencent.com", // 202.104.129.251 "sz4.tencent.com", // 202.104.129.254 "sz5.tencent.com", // 61.141.194.203 "sz6.tencent.com", // 202.104.129.252 "sz7.tencent.com", // 202.104.129.253 "202.96.170.64", "64.144.238.155", "202.104.129.254" }; const gint udp_server_amount = (sizeof(udp_server_list) / sizeof(udp_server_list[0])); const gchar *tcp_server_list[] = { "tcpconn.tencent.com", // 218.17.209.23 "tcpconn2.tencent.com", // 218.18.95.153 "tcpconn3.tencent.com", // 218.17.209.23 "tcpconn4.tencent.com", // 218.18.95.153 }; const gint tcp_server_amount = (sizeof(tcp_server_list) / sizeof(tcp_server_list[0])); static void _qq_login(GaimAccount * account) { const gchar *qq_server, *qq_port; qq_data *qd; GaimConnection *gc; GaimPresence *presence; gboolean login_hidden, use_tcp; g_return_if_fail(account != NULL); gc = gaim_account_get_connection(account); g_return_if_fail(gc != NULL); gc->flags |= GAIM_CONNECTION_HTML | GAIM_CONNECTION_NO_BGCOLOR | GAIM_CONNECTION_AUTO_RESP; qd = g_new0(qq_data, 1); gc->proto_data = qd; qq_server = gaim_account_get_string(account, "server", NULL); qq_port = gaim_account_get_string(account, "port", NULL); use_tcp = gaim_account_get_bool(account, "use_tcp", FALSE); presence = gaim_account_get_presence(account); login_hidden = gaim_presence_is_status_primitive_active(presence, GAIM_STATUS_INVISIBLE); qd->use_tcp = use_tcp; if (login_hidden) { qd->login_mode = QQ_LOGIN_MODE_HIDDEN; gaim_debug(GAIM_DEBUG_INFO, "QQ", "Login in hidden mode\n"); } else { qd->login_mode = QQ_LOGIN_MODE_NORMAL; gaim_debug(GAIM_DEBUG_INFO, "QQ", "Login in normal mode\n"); } if (qq_server == NULL || strlen(qq_server) == 0) qq_server = use_tcp ? tcp_server_list[random() % tcp_server_amount] : udp_server_list[random() % udp_server_amount]; if (qq_port == NULL || strtol(qq_port, NULL, 10) == 0) qq_port = use_tcp ? QQ_TCP_QUERY_PORT : QQ_UDP_PORT; gaim_connection_update_progress(gc, _("Connecting"), 0, QQ_CONNECT_STEPS); if (qq_connect(account, qq_server, strtol(qq_port, NULL, 10), use_tcp, FALSE) < 0) gaim_connection_error(gc, _("Unable to connect.")); } /* directly goes for qq_disconnect */ static void _qq_close(GaimConnection * gc) { g_return_if_fail(gc != NULL); qq_disconnect(gc); } /* returns the icon name for a buddy or protocol */ static const gchar *_qq_list_icon(GaimAccount * a, GaimBuddy * b) { /* XXX temp commented out until we figure out what to do with * status icons */ /* gchar *filename; qq_buddy *q_bud; gchar icon_suffix; */ /* do not use g_return_val_if_fail, as it is not assertion */ if (b == NULL || b->proto_data == NULL) return "qq"; /* q_bud = (qq_buddy *) b->proto_data; icon_suffix = get_suffix_from_status(q_bud->status); filename = get_icon_name(q_bud->icon / 3 + 1, icon_suffix); return filename; */ return "qq"; } /* a short status text beside buddy icon*/ static gchar *_qq_status_text(GaimBuddy *b) { qq_buddy *q_bud; GString *status; gchar *ret; q_bud = (qq_buddy *) b->proto_data; if (q_bud == NULL) return NULL; status = g_string_new(""); switch(q_bud->status) { case QQ_BUDDY_OFFLINE: g_string_append(status, "My Offline"); break; case QQ_BUDDY_ONLINE_NORMAL: return NULL; break; case QQ_BUDDY_ONLINE_OFFLINE: g_string_append(status, "Online Offline"); break; case QQ_BUDDY_ONLINE_AWAY: g_string_append(status, "Away"); break; case QQ_BUDDY_ONLINE_INVISIBLE: g_string_append(status, "Invisible"); break; default: g_string_printf(status, "Unknown-%d", q_bud->status); } /* switch (q_bud->gender) { case QQ_BUDDY_GENDER_GG: g_string_append(status, " GG"); break; case QQ_BUDDY_GENDER_MM: g_string_append(status, " MM"); break; case QQ_BUDDY_GENDER_UNKNOWN: g_string_append(status, "^_*"); break; default: g_string_append(status, "^_^"); } g_string_append_printf(status, " Age: %d", q_bud->age); g_string_append_printf(status, " Client: %04x", q_bud->client_version); */ // having_video = q_bud->comm_flag & QQ_COMM_FLAG_VIDEO; // if (having_video) // g_string_append(status, " (video)"); ret = status->str; g_string_free(status, FALSE); return ret; } /* a floating text when mouse is on the icon, show connection status here */ static void _qq_tooltip_text(GaimBuddy *b, GString *tooltip, gboolean full) { qq_buddy *q_bud; //gchar *country, *country_utf8, *city, *city_utf8; //guint32 ip_value; gchar *ip_str; g_return_if_fail(b != NULL); q_bud = (qq_buddy *) b->proto_data; //g_return_if_fail(q_bud != NULL); if (GAIM_BUDDY_IS_ONLINE(b) && q_bud != NULL) { /* ip_value = ntohl(*(guint32 *) (q_bud->ip)); if (qq_ip_get_location(ip_value, &country, &city)) { country_utf8 = qq_to_utf8(country, QQ_CHARSET_DEFAULT); city_utf8 = qq_to_utf8(city, QQ_CHARSET_DEFAULT); g_string_append_printf(tooltip, "\n%s, %s", country_utf8, city_utf8); g_free(country); g_free(city); g_free(country_utf8); g_free(city_utf8); } */ ip_str = gen_ip_str(q_bud->ip); if (strlen(ip_str) != 0) { g_string_append_printf(tooltip, "\n<b>%s Address:</b> %s:%d", (q_bud->comm_flag & QQ_COMM_FLAG_TCP_MODE) ? "TCP" : "UDP", ip_str, q_bud->port); } g_free(ip_str); g_string_append_printf(tooltip, "\n<b>Age:</b> %d", q_bud->age); switch (q_bud->gender) { case QQ_BUDDY_GENDER_GG: g_string_append(tooltip, "\n<b>Gender:</b> GG"); break; case QQ_BUDDY_GENDER_MM: g_string_append(tooltip, "\n<b>Gender:</b> MM"); break; case QQ_BUDDY_GENDER_UNKNOWN: g_string_append(tooltip, "\n<b>Gender:</b> UNKNOWN"); break; default: g_string_append_printf(tooltip, "\n<b>Gender:</b> ERROR(%d)", q_bud->gender); } /* switch gender */ g_string_append_printf(tooltip, "\n<b>Flag:</b> %01x", q_bud->flag1); g_string_append_printf(tooltip, "\n<b>CommFlag:</b> %01x", q_bud->comm_flag); g_string_append_printf(tooltip, "\n<b>Client:</b> %04x", q_bud->client_version); } } /* we can show tiny icons on the four corners of buddy icon, */ static void _qq_list_emblems(GaimBuddy * b, const char **se, const char **sw, const char **nw, const char **ne) { // each char ** are refering to filename in pixmaps/gaim/status/default/*png qq_buddy *q_bud = b->proto_data; const char *emblems[4] = { NULL, NULL, NULL, NULL }; int i = 0; if (q_bud == NULL) { emblems[0] = "offline"; } else { if (q_bud->comm_flag & QQ_COMM_FLAG_QQ_MEMBER) emblems[i++] = "qq_member"; if (q_bud->comm_flag & QQ_COMM_FLAG_BIND_MOBILE) emblems[i++] = "wireless"; if (q_bud->comm_flag & QQ_COMM_FLAG_VIDEO) emblems[i++] = "video"; } *se = emblems[0]; *sw = emblems[1]; *nw = emblems[2]; *ne = emblems[3]; return; } /* QQ away status (used to initiate QQ away packet) */ static GList *_qq_away_states(GaimAccount *ga) { GaimStatusType *status; GList *types = NULL; status = gaim_status_type_new_full(GAIM_STATUS_AVAILABLE, "available", _("QQ: Available"), FALSE, TRUE, FALSE); types = g_list_append(types, status); status = gaim_status_type_new_full(GAIM_STATUS_AWAY, "away", _("QQ: Away"), FALSE, TRUE, FALSE); types = g_list_append(types, status); status = gaim_status_type_new_full(GAIM_STATUS_INVISIBLE, "invisible", _("QQ: Invisible"), FALSE, TRUE, FALSE); types = g_list_append(types, status); status = gaim_status_type_new_full(GAIM_STATUS_OFFLINE, "offline", _("QQ: Offline"), FALSE, TRUE, FALSE); types = g_list_append(types, status); return types; } /* initiate QQ away with proper change_status packet */ static void _qq_set_away(GaimAccount *account, GaimStatus *status) { GaimConnection *gc = gaim_account_get_connection(account); const char *state = gaim_status_get_id(status); qq_data *qd; g_return_if_fail(gc != NULL && gc->proto_data != NULL); qd = (qq_data *) gc->proto_data; if(0 == strcmp(state, "available")) qd->status = QQ_SELF_STATUS_AVAILABLE; else if (0 == strcmp(state, "away")) qd->status = QQ_SELF_STATUS_AWAY; else if (0 == strcmp(state, "invisible")) qd->status = QQ_SELF_STATUS_INVISIBLE; else qd->status = QQ_SELF_STATUS_AVAILABLE; qq_send_packet_change_status(gc); } // IMPORTANT: GaimConvImFlags -> GaimMessageFlags /* send an instance msg to a buddy */ static gint _qq_send_im(GaimConnection * gc, const gchar * who, const gchar * message, GaimMessageFlags flags) { gint type, to_uid; gchar *msg, *msg_with_qq_smiley; qq_data *qd; g_return_val_if_fail(gc != NULL && gc->proto_data != NULL && who != NULL, -1); qd = (qq_data *) gc->proto_data; g_return_val_if_fail(strlen(message) <= QQ_MSG_IM_MAX, -E2BIG); type = (flags == GAIM_MESSAGE_AUTO_RESP ? QQ_IM_AUTO_REPLY : QQ_IM_TEXT); to_uid = gaim_name_to_uid(who); /* if msg is to myself, bypass the network */ if (to_uid == qd->uid) serv_got_im(gc, who, message, flags, time(NULL)); else { msg = utf8_to_qq(message, QQ_CHARSET_DEFAULT); msg_with_qq_smiley = gaim_smiley_to_qq(msg); qq_send_packet_im(gc, to_uid, msg_with_qq_smiley, type); g_free(msg); g_free(msg_with_qq_smiley); } return 1; } /* send a chat msg to a QQ Qun */ static int _qq_chat_send(GaimConnection *gc, int channel, const char *message, GaimMessageFlags flags) { gchar *msg, *msg_with_qq_smiley; qq_group *group; g_return_val_if_fail(gc != NULL && message != NULL, -1); g_return_val_if_fail(strlen(message) <= QQ_MSG_IM_MAX, -E2BIG); group = qq_group_find_by_channel(gc, channel); g_return_val_if_fail(group != NULL, -1); msg = utf8_to_qq(message, QQ_CHARSET_DEFAULT); msg_with_qq_smiley = gaim_smiley_to_qq(msg); qq_send_packet_group_im(gc, group, msg_with_qq_smiley); g_free(msg); g_free(msg_with_qq_smiley); return 1; } /* send packet to get who's detailed information */ static void _qq_get_info(GaimConnection * gc, const gchar * who) { guint32 uid; qq_data *qd; g_return_if_fail(gc != NULL && gc->proto_data != NULL); qd = gc->proto_data; uid = gaim_name_to_uid(who); if (uid <= 0) { gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Not valid QQid: %s\n", who); gaim_notify_error(gc, NULL, _("Invalid name, please input in qq-xxxxxxxx format"), NULL); return; } qq_send_packet_get_info(gc, uid, TRUE); } /* get my own information */ static void _qq_menu_modify_my_info(GaimPluginAction * action) { GaimConnection *gc = (GaimConnection *) action->context; qq_data *qd; g_return_if_fail(gc != NULL && gc->proto_data != NULL); qd = (qq_data *) gc->proto_data; //_qq_get_info(gc, uid_to_gaim_name(qd->uid)); qq_prepare_modify_info(gc); } static void _qq_menu_change_password(GaimPluginAction * action) { gaim_notify_uri(NULL, "https://password.qq.com"); } /* remove a buddy from my list and remove myself from his list */ /* TODO: re-enable this static void _qq_menu_block_buddy(GaimBlistNode * node) { guint32 uid; gc_and_uid *g; GaimBuddy *buddy; GaimConnection *gc; // const gchar *who = param_who; gfhuang const gchar *who; g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(node)); buddy = (GaimBuddy *) node; gc = gaim_account_get_connection(buddy->account); who = buddy->name; g_return_if_fail(gc != NULL && who != NULL); uid = gaim_name_to_uid(who); g_return_if_fail(uid > 0); g = g_new0(gc_and_uid, 1); g->gc = gc; g->uid = uid; gaim_request_action(gc, _("Block Buddy"), _("Are you sure to block this buddy?"), NULL, 1, g, 2, _("Cancel"), G_CALLBACK(qq_do_nothing_with_gc_and_uid), _("Block"), G_CALLBACK(qq_block_buddy_with_gc_and_uid)); } */ /* show a brief summary of what we get from login packet */ static void _qq_menu_show_login_info(GaimPluginAction * action) { GaimConnection *gc = (GaimConnection *) action->context; qq_data *qd; GString *info; g_return_if_fail(gc != NULL && gc->proto_data != NULL); qd = (qq_data *) gc->proto_data; info = g_string_new("<html><body>\n"); g_string_append_printf(info, _("<b>Current Online</b>: %d<br>\n"), qd->all_online); g_string_append_printf(info, _("<b>Last Refresh</b>: %s<br>\n"), ctime(&qd->last_get_online)); g_string_append(info, "<hr>\n"); g_string_append_printf(info, _("<b>Connection Mode</b>: %s<br>\n"), qd->use_tcp ? "TCP" : "UDP"); g_string_append_printf(info, _("<b>Server IP</b>: %s: %d<br>\n"), qd->server_ip, qd->server_port); g_string_append_printf(info, _("<b>My Public IP</b>: %s<br>\n"), qd->my_ip); g_string_append(info, "<hr>\n"); g_string_append(info, "<i>Information below may not be accurate</i><br>\n"); g_string_append_printf(info, _("<b>Login Time</b>: %s<br>\n"), ctime(&qd->login_time)); g_string_append_printf(info, _("<b>Last Login IP</b>: %s<br>\n"), qd->last_login_ip); g_string_append_printf(info, _("<b>Last Login Time</b>: %s\n"), ctime(&qd->last_login_time)); g_string_append(info, "</body></html>"); gaim_notify_formatted(gc, NULL, _("Login Information"), NULL, info->str, NULL, NULL); g_string_free(info, TRUE); } /* static void _qq_menu_locate_ip_cb(GaimConnection * gc, GaimRequestFields * fields) { GList *groups, *flds; GaimRequestField *field; const gchar *id, *value; gchar *ip_str = NULL, *ip_dupstr = NULL; guint8 *ip; gchar *country, *country_utf8, *city, *city_utf8; guint32 ip_value; for (groups = gaim_request_fields_get_groups(fields); groups && !ip_str; groups = groups->next) { for (flds = gaim_request_field_group_get_fields(groups->data); flds && !ip_str; flds = flds->next) { field = flds->data; id = gaim_request_field_get_id(field); value = gaim_request_field_string_get_value(field); if (!g_ascii_strcasecmp(id, "ip")) { ip_str = g_strdup(value); break; } } } if(ip_str) { ip = str_ip_gen(ip_str); ip_dupstr = gen_ip_str(ip); ip_value = ntohl(*(guint32 *)ip); if (qq_ip_get_location(ip_value, &country, &city)) { country_utf8 = qq_to_utf8(country, QQ_CHARSET_DEFAULT); city_utf8 = qq_to_utf8(city, QQ_CHARSET_DEFAULT); gaim_notify_info(gc, ip_dupstr, country_utf8, city_utf8); g_free(country); g_free(city); g_free(country_utf8); g_free(city_utf8); } else gaim_notify_info(gc, ip_dupstr, "IP not found", NULL); g_free(ip); g_free(ip_dupstr); g_free(ip_str); } } static void _qq_menu_locate_ip(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; GaimRequestField *field; GaimRequestFields *fields; GaimRequestFieldGroup *group; g_return_if_fail(gc != NULL); fields = gaim_request_fields_new(); group = gaim_request_field_group_new(NULL); gaim_request_fields_add_group(fields, group); field = gaim_request_field_string_new("ip", _("IP Address"), NULL, FALSE); gaim_request_field_group_add_field(group, field); gaim_request_fields(gc, _("Locate an IP"), _("Locate an IP address"), NULL, fields, _("Check"), G_CALLBACK(_qq_menu_locate_ip_cb), _("Cancel"), NULL, gc); } */ /* static void _qq_menu_search_or_add_permanent_group(GaimPluginAction * action) { gaim_gtk_roomlist_dialog_show(); } */ /* static void _qq_menu_create_permanent_group(GaimPluginAction * action) { GaimConnection *gc = (GaimConnection *) action->context; g_return_if_fail(gc != NULL); gaim_request_input(gc, _("Create QQ Qun"), _("Input Qun name here"), _("Only QQ member can create permanent Qun"), "OpenQ", FALSE, FALSE, NULL, _("Create"), G_CALLBACK(qq_group_create_with_name), _("Cancel"), NULL, gc); } */ /* XXX re-enable this static void _qq_menu_unsubscribe_group(GaimBlistNode * node) { GaimChat *chat = (GaimChat *)node; GaimConnection *gc = gaim_account_get_connection(chat->account); GHashTable *components = chat -> components; g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); g_return_if_fail(gc != NULL && components != NULL); qq_group_exit(gc, components); } // XXX re-enable this static void _qq_menu_manage_group(GaimBlistNode * node) { GaimChat *chat = (GaimChat *)node; GaimConnection *gc = gaim_account_get_connection(chat->account); GHashTable *components = chat -> components; g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node)); g_return_if_fail(gc != NULL && components != NULL); qq_group_manage_group(gc, components); } */ /* static void _qq_menu_show_system_message(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; g_return_if_fail ( gc != NULL ); gaim_gtk_log_show(GAIM_LOG_IM, "systemim", gaim_connection_get_account(gc)); } */ /* TODO: re-enable this static void _qq_menu_send_file(GaimBlistNode * node, gpointer ignored) { GaimBuddy *buddy; GaimConnection *gc; qq_buddy *q_bud; g_return_if_fail (GAIM_BLIST_NODE_IS_BUDDY (node)); buddy = (GaimBuddy *) node; q_bud = (qq_buddy *) buddy->proto_data; // if (is_online (q_bud->status)) { gc = gaim_account_get_connection (buddy->account); g_return_if_fail (gc != NULL && gc->proto_data != NULL); qq_send_file(gc, buddy->name, NULL); // } } */ /* static void _qq_send_custom_packet(GaimConnection *gc, const gchar *packet) { guint16 cmd; guint8 *buffer; gint len; if (!packet) { gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Null packet inputted!\n"); return; } if (strlen(packet) > MAX_PACKET_SIZE * 2) { gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Packet inputted is too large!\n"); return; } if (strlen(packet) < 4) { gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Packet is impossibly short!\n"); return; } buffer = hex_str_to_bytes(packet); if (!buffer) { gaim_debug(GAIM_DEBUG_ERROR, "QQ", "Invalid packet inputted!\n"); return; } // big endian cmd = 256 * buffer[0] + buffer[1]; gaim_debug(GAIM_DEBUG_INFO, "QQ", "Inputted CMD: %d\n", cmd); len = strlen(buffer) - 2; packet = buffer + 2; qq_send_cmd(gc, cmd, TRUE, 0, TRUE, packet, len); g_free(buffer); } */ /* send a custom packet to the server - for protocol testing */ /* static void _qq_menu_send_custom_packet(GaimPluginAction *action) { GaimConnection *gc = (GaimConnection *) action->context; g_return_if_fail(gc != NULL); gaim_request_input(gc, _("Send Custom Packet"), _("Enter the packet in hex here"), _("Include the command and everything following"), NULL, FALSE, FALSE, NULL, _("Send"), G_CALLBACK(_qq_send_custom_packet), _("Cancel"), NULL, gc); } */ /* protocol related menus */ static GList *_qq_actions(GaimPlugin * plugin, gpointer context) { GList *m; GaimPluginAction *act; m = NULL; act = gaim_plugin_action_new(_("Modify My Information"), _qq_menu_modify_my_info); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Change Password"), _qq_menu_change_password); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Show Login Information"), _qq_menu_show_login_info); m = g_list_append(m, act); /* act = gaim_plugin_action_new(_("Send Custom Packet"), _qq_menu_send_custom_packet); m = g_list_append(m, act); */ /* XXX consider re-enabling this act = gaim_plugin_action_new(_("Show System Message"), _qq_menu_show_system_message); m = g_list_append(m, act); */ /* XXX the old group gtk code needs to moved to the gaim UI before this can be used act = gaim_plugin_action_new(_("Qun: Search a permanent Qun"), _qq_menu_search_or_add_permanent_group); m = g_list_append(m, act); act = gaim_plugin_action_new(_("Qun: Create a permanent Qun"), _qq_menu_create_permanent_group); m = g_list_append(m, act); */ /* XXX consider re-enabling this act = gaim_plugin_action_new(_("Locate an IP"), _qq_menu_locate_ip); m = g_list_append(m, act); */ return m; } /* chat-related (QQ Qun) menu shown up with right-click */ /* XXX re-enable this static GList *_qq_chat_menu(GaimBlistNode *node) { GList *m; GaimMenuAction *act; m = NULL; act = gaim_menu_action_new(_("Exit this QQ Qun"), GAIM_CALLBACK(_qq_menu_unsubscribe_group), NULL, NULL); m = g_list_append(m, act); act = gaim_menu_action_new(_("Show Details"), GAIM_CALLBACK(_qq_menu_manage_group), NULL, NULL); m = g_list_append(m, act); return m; } */ /* buddy-related menu shown up with right-click */ /* XXX re-enable this static GList *_qq_buddy_menu(GaimBlistNode * node) { GList *m; if(GAIM_BLIST_NODE_IS_CHAT(node)) return _qq_chat_menu(node); m = NULL; */ /* TODO : not working, temp commented out by gfhuang act = gaim_menu_action_new(_("Block this buddy"), GAIM_CALLBACK(_qq_menu_block_buddy), NULL, NULL); //add NULL by gfhuang m = g_list_append(m, act); // if (q_bud && is_online(q_bud->status)) { act = gaim_menu_action_new(_("Send File"), GAIM_CALLBACK(_qq_menu_send_file), NULL, NULL); //add NULL by gfhuang m = g_list_append(m, act); // } */ /* return m; } */ static void _qq_keep_alive(GaimConnection * gc) { qq_group *group; qq_data *qd; GList *list; g_return_if_fail(gc != NULL); if (NULL == (qd = (qq_data *) gc->proto_data)) return; list = qd->groups; while (list != NULL) { group = (qq_group *) list->data; if (group->my_status == QQ_GROUP_MEMBER_STATUS_IS_MEMBER || group->my_status == QQ_GROUP_MEMBER_STATUS_IS_ADMIN) // no need to get info time and time again, online members enough qq_send_cmd_group_get_online_member(gc, group); list = list->next; } qq_send_packet_keep_alive(gc); } /* convert chat nickname to qq-uid to get this buddy info */ /* who is the nickname of buddy in QQ chat-room (Qun) */ static void _qq_get_chat_buddy_info(GaimConnection * gc, gint channel, const gchar * who) { gchar *gaim_name; g_return_if_fail(gc != NULL && gc->proto_data != NULL && who != NULL); gaim_name = qq_group_find_member_by_channel_and_nickname(gc, channel, who); if (gaim_name != NULL) _qq_get_info(gc, gaim_name); } /* convert chat nickname to qq-uid to invite individual IM to buddy */ /* who is the nickname of buddy in QQ chat-room (Qun) */ static gchar *_qq_get_chat_buddy_real_name(GaimConnection * gc, gint channel, const gchar * who) { g_return_val_if_fail(gc != NULL && gc->proto_data != NULL && who != NULL, NULL); return qq_group_find_member_by_channel_and_nickname(gc, channel, who); } void qq_function_not_implemented(GaimConnection * gc) { gaim_notify_warning(gc, NULL, _("This function has not be implemented yet"), _("Please wait for new version")); } GaimPlugin *my_protocol = NULL; static GaimPluginProtocolInfo prpl_info = { OPT_PROTO_CHAT_TOPIC | OPT_PROTO_USE_POINTSIZE, NULL, /* user_splits */ NULL, /* protocol_options */ NO_BUDDY_ICONS, /* icon_spec */ _qq_list_icon, /* list_icon */ _qq_list_emblems, /* list_emblems */ _qq_status_text, /* status_text */ _qq_tooltip_text, /* tooltip_text */ _qq_away_states, /* away_states */ NULL, /* blist_node_menu */ qq_chat_info, /* chat_info */ NULL, /* chat_info_defaults */ _qq_login, /* login */ _qq_close, /* close */ _qq_send_im, /* send_im */ NULL, /* set_info */ NULL, /* send_typing */ _qq_get_info, /* get_info */ _qq_set_away, /* set_away */ NULL, /* set_idle */ NULL, /* change_passwd */ qq_add_buddy, /* add_buddy */ NULL, /* add_buddies */ qq_remove_buddy, /* remove_buddy */ NULL, /* remove_buddies */ NULL, /* add_permit */ NULL, /* add_deny */ NULL, /* rem_permit */ NULL, /* rem_deny */ NULL, /* set_permit_deny */ qq_group_join, /* join_chat */ NULL, /* reject chat invite */ NULL, /* get_chat_name */ NULL, /* chat_invite */ NULL, /* chat_leave */ NULL, /* chat_whisper */ _qq_chat_send, /* chat_send */ _qq_keep_alive, /* keepalive */ NULL, /* register_user */ _qq_get_chat_buddy_info, /* get_cb_info */ NULL, /* get_cb_away */ NULL, /* alias_buddy */ NULL, /* group_buddy */ NULL, /* rename_group */ NULL, /* buddy_free */ NULL, /* convo_closed */ NULL, /* normalize */ NULL, /* set_buddy_icon */ NULL, /* remove_group */ _qq_get_chat_buddy_real_name, /* get_cb_real_name */ NULL, /* set_chat_topic */ NULL, /* find_blist_chat */ qq_roomlist_get_list, /* roomlist_get_list */ qq_roomlist_cancel, /* roomlist_cancel */ NULL, /* roomlist_expand_category */ NULL, /* can_receive_file */ qq_send_file, /* send_file */ NULL, /* new xfer */ NULL, /* offline_message */ NULL, /* GaimWhiteboardPrplOps */ }; static GaimPluginInfo info = { GAIM_PLUGIN_MAGIC, GAIM_MAJOR_VERSION, GAIM_MINOR_VERSION, GAIM_PLUGIN_PROTOCOL, /**< type */ NULL, /**< ui_requirement */ 0, /**< flags */ NULL, /**< dependencies */ GAIM_PRIORITY_DEFAULT, /**< priority */ "prpl-qq", /**< id */ "QQ", /**< name */ VERSION, /**< version */ /** summary */ N_("QQ Protocol Plugin"), /** description */ N_("QQ Protocol Plugin"), OPENQ_AUTHOR, /**< author */ OPENQ_WEBSITE, /**< homepage */ NULL, /**< load */ NULL, /**< unload */ NULL, /**< destroy */ NULL, /**< ui_info */ &prpl_info, /**< extra_info */ NULL, /**< prefs_info */ _qq_actions }; static void init_plugin(GaimPlugin * plugin) { GaimAccountOption *option; bindtextdomain(PACKAGE, LOCALEDIR); bind_textdomain_codeset(PACKAGE, "UTF-8"); option = gaim_account_option_bool_new(_("Login in TCP"), "use_tcp", FALSE); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); option = gaim_account_option_bool_new(_("Login Hidden"), "hidden", FALSE); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); option = gaim_account_option_string_new(_("QQ Server"), "server", NULL); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); option = gaim_account_option_string_new(_("QQ Port"), "port", NULL); prpl_info.protocol_options = g_list_append(prpl_info.protocol_options, option); my_protocol = plugin; gaim_prefs_add_none("/plugins/prpl/qq"); gaim_prefs_add_string("/plugins/prpl/qq/ipfile", ""); gaim_prefs_add_bool("/plugins/prpl/qq/show_status_by_icon", TRUE); gaim_prefs_add_bool("/plugins/prpl/qq/show_fake_video", FALSE); gaim_prefs_add_string("/plugins/prpl/qq/datadir", DATADIR); gaim_prefs_add_bool("/plugins/prpl/qq/prompt_for_missing_packet", TRUE); gaim_prefs_add_bool("/plugins/prpl/qq/prompt_group_msg_on_recv", TRUE); } GAIM_INIT_PLUGIN(qq, init_plugin, info);