14192
|
1 /**
|
|
2 * The QQ2003C protocol plugin
|
|
3 *
|
|
4 * for gaim
|
|
5 *
|
|
6 * Copyright (C) 2004 Puzzlebird
|
|
7 *
|
|
8 * This program is free software; you can redistribute it and/or modify
|
|
9 * it under the terms of the GNU General Public License as published by
|
|
10 * the Free Software Foundation; either version 2 of the License, or
|
|
11 * (at your option) any later version.
|
|
12 *
|
|
13 * This program is distributed in the hope that it will be useful,
|
|
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
16 * GNU General Public License for more details.
|
|
17 *
|
|
18 * You should have received a copy of the GNU General Public License
|
|
19 * along with this program; if not, write to the Free Software
|
|
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
21 */
|
|
22
|
|
23 #include "debug.h"
|
|
24 #include "internal.h"
|
|
25
|
|
26 #include "char_conv.h"
|
|
27 #include "packet_parse.h"
|
|
28 #include "qq.h"
|
|
29 #include "utils.h"
|
|
30
|
|
31 #define QQ_SMILEY_AMOUNT 96
|
|
32
|
|
33 #define UTF8 "UTF-8"
|
|
34 #define QQ_CHARSET_ZH_CN "GBK"
|
|
35 #define QQ_CHARSET_ENG "ISO-8859-1"
|
|
36
|
|
37 #define QQ_NULL_MSG "(NULL)" /* return this if conversion fails */
|
|
38 #define QQ_NULL_SMILEY "(SM)" /* return this if smiley conversion fails */
|
|
39
|
|
40 /* a debug function */
|
|
41 void _qq_show_packet(const gchar *desc, const guint8 *buf, gint len);
|
|
42
|
|
43 const gchar qq_smiley_map[QQ_SMILEY_AMOUNT] = {
|
|
44 0x41, 0x43, 0x42, 0x44, 0x45, 0x46, 0x47, 0x48,
|
|
45 0x49, 0x4a, 0x4b, 0x4c, 0x4d, 0x4e, 0x4f, 0x73,
|
|
46 0x74, 0x75, 0x76, 0x77, 0x8a, 0x8b, 0x8c, 0x8d,
|
|
47 0x8e, 0x8f, 0x78, 0x79, 0x7a, 0x7b, 0x90, 0x91,
|
|
48 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99,
|
|
49 0x59, 0x5a, 0x5c, 0x58, 0x57, 0x55, 0x7c, 0x7d,
|
|
50 0x7e, 0x7f, 0x9a, 0x9b, 0x60, 0x67, 0x9c, 0x9d,
|
|
51 0x9e, 0x5e, 0x9f, 0x89, 0x80, 0x81, 0x82, 0x62,
|
|
52 0x63, 0x64, 0x65, 0x66, 0x83, 0x68, 0x84, 0x85,
|
|
53 0x86, 0x87, 0x6b, 0x6e, 0x6f, 0x70, 0x88, 0xa0,
|
|
54 0x50, 0x51, 0x52, 0x53, 0x54, 0x56, 0x5b, 0x5d,
|
|
55 0x5f, 0x61, 0x69, 0x6a, 0x6c, 0x6d, 0x71, 0x72
|
|
56 };
|
|
57
|
|
58
|
|
59 const gchar *gaim_smiley_map[QQ_SMILEY_AMOUNT] = {
|
|
60 "/jy", "/pz", "/se", "/fd", "/dy", "/ll", "/hx", "/bz",
|
|
61 "/shui", "/dk ", "/gg", "/fn", "/tp", "/cy", "/wx", "/ng",
|
|
62 "/kuk", "/feid", "/zk", "/tu", "/tx", "/ka", "/by", "/am",
|
|
63 "/jie", "/kun", "/jk", "/lh", "/hanx", "/db", "/fendou",
|
|
64 "/zhm",
|
|
65 "/yiw", "/xu", "/yun", "/zhem", "/shuai", "/kl", "/qiao",
|
|
66 "/zj",
|
|
67 "/shan", "/fad", "/aiq", "/tiao", "/zhao", "/mm", "/zt",
|
|
68 "/maom",
|
|
69 "/xg", "/yb", "/qianc", "/dp", "/bei", "/dg", "/shd",
|
|
70 "/zhd",
|
|
71 "/dao", "/zq", "/yy", "/bb", "/gf", "/fan", "/yw", "/mg",
|
|
72 "/dx", "/wen", "/xin", "/xs", "/hy", "/lw", "/dh", "/sj",
|
|
73 "/yj", "/ds", "/ty", "/yl", "/qiang", "/ruo", "/ws",
|
|
74 "/shl",
|
|
75 "/dd", "/mn", "/hl", "/mamao", "/qz", "/fw", "/oh", "/bj",
|
|
76 "/qsh", "/xig", "/xy", "/duoy", "/xr", "/xixing", "/nv",
|
|
77 "/nan"
|
|
78 };
|
|
79
|
|
80 /* these functions parse font-attr */
|
|
81 static gchar _get_size(gchar font_attr)
|
|
82 {
|
|
83 return font_attr & 0x1f;
|
|
84 }
|
|
85
|
|
86 static gboolean _check_bold(gchar font_attr)
|
|
87 {
|
|
88 return (font_attr & 0x20) ? TRUE : FALSE;
|
|
89 }
|
|
90
|
|
91 static gboolean _check_italic(gchar font_attr)
|
|
92 {
|
|
93 return (font_attr & 0x40) ? TRUE : FALSE;
|
|
94 }
|
|
95
|
|
96 static gboolean _check_underline(gchar font_attr)
|
|
97 {
|
|
98 return (font_attr & 0x80) ? TRUE : FALSE;
|
|
99 }
|
|
100
|
|
101 /* convert a string from from_charset to to_charset, using g_convert */
|
|
102 static gchar *_my_convert(const gchar *str, gssize len, const gchar *to_charset, const gchar *from_charset)
|
|
103 {
|
|
104 GError *error = NULL;
|
|
105 gchar *ret;
|
|
106 gsize byte_read, byte_write;
|
|
107
|
|
108 g_return_val_if_fail(str != NULL && to_charset != NULL && from_charset != NULL, g_strdup(QQ_NULL_MSG));
|
|
109
|
|
110 ret = g_convert(str, len, to_charset, from_charset, &byte_read, &byte_write, &error);
|
|
111
|
|
112 if (error == NULL)
|
|
113 return ret; /* conversion is OK */
|
|
114 else { /* conversion error */
|
|
115 gaim_debug(GAIM_DEBUG_ERROR, "QQ", "%s\n", error->message);
|
|
116 gaim_debug(GAIM_DEBUG_WARNING, "QQ",
|
|
117 "Dump failed text\n%s",
|
|
118 hex_dump_to_str((guint8 *) str, (len == -1) ? strlen(str) : len));
|
|
119 g_error_free(error);
|
|
120 return g_strdup(QQ_NULL_MSG);
|
|
121 }
|
|
122 }
|
|
123
|
|
124 /* take the input as a pascal string and return a converted c-string in UTF-8
|
|
125 * returns the number of bytes read, return -1 if fatal error
|
|
126 * the converted UTF-8 will be saved in ret */
|
|
127 gint convert_as_pascal_string(guint8 *data, gchar **ret, const gchar *from_charset)
|
|
128 {
|
|
129 guint8 len;
|
|
130
|
|
131 g_return_val_if_fail(data != NULL && from_charset != NULL, -1);
|
|
132
|
|
133 len = data[0];
|
|
134 *ret = _my_convert((gchar *) (data + 1), (gssize) len, UTF8, from_charset);
|
|
135
|
|
136 return len + 1;
|
|
137 }
|
|
138
|
14347
|
139 /* convert QQ formatted msg to Gaim formatted msg (and UTF-8) */
|
14192
|
140 gchar *qq_encode_to_gaim(guint8 *data, gint len, const gchar *msg)
|
|
141 {
|
|
142 GString *encoded;
|
|
143 guint8 font_attr, font_size, color[3], bar, *cursor;
|
|
144 gboolean is_bold, is_italic, is_underline;
|
|
145 guint16 charset_code;
|
14347
|
146 gchar *font_name, *color_code, *msg_utf8, *tmp, *ret;
|
14192
|
147
|
|
148 cursor = data;
|
|
149 _qq_show_packet("QQ_MESG recv for font style", data, len);
|
|
150
|
|
151 read_packet_b(data, &cursor, len, &font_attr);
|
|
152 read_packet_data(data, &cursor, len, color, 3); /* red,green,blue */
|
|
153 color_code = g_strdup_printf("#%02x%02x%02x", color[0], color[1], color[2]);
|
|
154
|
|
155 read_packet_b(data, &cursor, len, &bar); /* skip, not sure of its use */
|
|
156 read_packet_w(data, &cursor, len, &charset_code);
|
|
157
|
14347
|
158 tmp = g_strndup((gchar *) cursor, data + len - cursor);
|
|
159 font_name = qq_to_utf8(tmp, QQ_CHARSET_DEFAULT);
|
|
160 g_free(tmp);
|
14192
|
161
|
|
162 font_size = _get_size(font_attr);
|
|
163 is_bold = _check_bold(font_attr);
|
|
164 is_italic = _check_italic(font_attr);
|
|
165 is_underline = _check_underline(font_attr);
|
|
166
|
|
167 /* Although there is charset returned from QQ msg, it can't be used.
|
|
168 * For example, if a user send a Chinese message from English Windows
|
|
169 * the charset_code in QQ msg is 0x0000, not 0x8602.
|
|
170 * Therefore, it is better to use uniform conversion.
|
|
171 * By default, we use GBK, which includes all character of SC, TC, and EN. */
|
|
172 msg_utf8 = qq_to_utf8(msg, QQ_CHARSET_DEFAULT);
|
|
173 encoded = g_string_new("");
|
|
174
|
|
175 /* Henry: The range QQ sends rounds from 8 to 22, where a font size
|
|
176 * of 10 is equal to 3 in html font tag */
|
|
177 g_string_append_printf(encoded,
|
|
178 "<font color=\"%s\"><font face=\"%s\"><font size=\"%d\">",
|
|
179 color_code, font_name, font_size / 3);
|
|
180 gaim_debug(GAIM_DEBUG_INFO, "QQ_MESG",
|
|
181 "recv <font color=\"%s\"><font face=\"%s\"><font size=\"%d\">\n",
|
|
182 color_code, font_name, font_size / 3);
|
|
183 g_string_append(encoded, msg_utf8);
|
|
184
|
|
185 if (is_bold) {
|
|
186 g_string_prepend(encoded, "<b>");
|
|
187 g_string_append(encoded, "</b>");
|
|
188 }
|
|
189 if (is_italic) {
|
|
190 g_string_prepend(encoded, "<i>");
|
|
191 g_string_append(encoded, "</i>");
|
|
192 }
|
|
193 if (is_underline) {
|
|
194 g_string_prepend(encoded, "<u>");
|
|
195 g_string_append(encoded, "</u>");
|
|
196 }
|
|
197
|
|
198 g_string_append(encoded, "</font></font></font>");
|
|
199 ret = encoded->str;
|
|
200
|
|
201 g_free(msg_utf8);
|
|
202 g_free(font_name);
|
|
203 g_free(color_code);
|
|
204 g_string_free(encoded, FALSE);
|
|
205
|
|
206 return ret;
|
|
207 }
|
|
208
|
|
209 /* two convenience methods, using _my_convert */
|
|
210 gchar *utf8_to_qq(const gchar *str, const gchar *to_charset)
|
|
211 {
|
|
212 return _my_convert(str, -1, to_charset, UTF8);
|
|
213 }
|
|
214
|
|
215 gchar *qq_to_utf8(const gchar *str, const gchar *from_charset)
|
|
216 {
|
|
217 return _my_convert(str, -1, UTF8, from_charset);
|
|
218 }
|
|
219
|
|
220 /* QQ uses binary code for smiley, while gaim uses strings.
|
|
221 * There is a mapping relation between these two. */
|
|
222 gchar *qq_smiley_to_gaim(gchar *text)
|
|
223 {
|
|
224 gint index;
|
|
225 gchar qq_smiley, *cur_seg, **segments, *ret;
|
|
226 GString *converted;
|
|
227
|
|
228 converted = g_string_new("");
|
|
229 segments = split_data((guint8 *) text, strlen(text), "\x14", 0);
|
|
230 g_string_append(converted, segments[0]);
|
|
231
|
|
232 while ((*(++segments)) != NULL) {
|
|
233 cur_seg = *segments;
|
|
234 qq_smiley = cur_seg[0];
|
|
235 for (index = 0; index < QQ_SMILEY_AMOUNT; index++) {
|
|
236 if (qq_smiley_map[index] == qq_smiley)
|
|
237 break;
|
|
238 }
|
|
239 if (index >= QQ_SMILEY_AMOUNT) {
|
|
240 g_string_append(converted, QQ_NULL_SMILEY);
|
|
241 } else {
|
|
242 g_string_append(converted, gaim_smiley_map[index]);
|
|
243 g_string_append(converted, (cur_seg + 1));
|
|
244 }
|
|
245 }
|
|
246
|
|
247 ret = converted->str;
|
|
248 g_string_free(converted, FALSE);
|
|
249 return ret;
|
|
250 }
|
|
251
|
|
252 /* convert smiley from gaim style to qq binary code */
|
|
253 gchar *gaim_smiley_to_qq(gchar *text)
|
|
254 {
|
|
255 gchar *begin, *cursor, *ret;
|
|
256 gint index;
|
|
257 GString *converted;
|
|
258
|
|
259 converted = g_string_new(text);
|
|
260
|
|
261 for (index = 0; index < QQ_SMILEY_AMOUNT; index++) {
|
|
262 begin = cursor = converted->str;
|
|
263 while ((cursor = g_strstr_len(cursor, -1, gaim_smiley_map[index]))) {
|
|
264 g_string_erase(converted, (cursor - begin), strlen(gaim_smiley_map[index]));
|
|
265 g_string_insert_c(converted, (cursor - begin), 0x14);
|
|
266 g_string_insert_c(converted, (cursor - begin + 1), qq_smiley_map[index]);
|
|
267 cursor++;
|
|
268 }
|
|
269 }
|
|
270 g_string_append_c(converted, 0x20); /* important for last smiiley */
|
|
271
|
|
272 ret = converted->str;
|
|
273 g_string_free(converted, FALSE);
|
|
274 return ret;
|
|
275 }
|