Mercurial > pidgin.yaz
comparison src/gtkimhtml.c @ 1428:00b3d02a2168
[gaim-migrate @ 1438]
gtkhtml has gotten replaced by gtkimhtml.
committer: Tailor Script <tailor@pidgin.im>
author | Eric Warmenhoven <eric@warmenhoven.org> |
---|---|
date | Fri, 26 Jan 2001 02:02:36 +0000 |
parents | |
children | 5df631739769 |
comparison
equal
deleted
inserted
replaced
1427:28278bd61403 | 1428:00b3d02a2168 |
---|---|
1 /* | |
2 * GtkIMHtml | |
3 * | |
4 * Copyright (C) 2000, Eric Warmenhoven <warmenhoven@yahoo.com> | |
5 * | |
6 * This program is free software; you can redistribute it and/or modify | |
7 * under the terms of the GNU General Public License as published by | |
8 * the Free Software Foundation; either version 2 of the License, or | |
9 * (at your option) any later version. | |
10 * | |
11 * This program is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License | |
17 * along with this program; if not, write to the Free Software | |
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | |
19 * | |
20 */ | |
21 | |
22 #include "gtkimhtml.h" | |
23 #include <gtk/gtk.h> | |
24 #include <string.h> | |
25 #include <ctype.h> | |
26 #include <stdio.h> | |
27 #include <math.h> | |
28 | |
29 #include "pixmaps/angel.xpm" | |
30 #include "pixmaps/bigsmile.xpm" | |
31 #include "pixmaps/burp.xpm" | |
32 #include "pixmaps/crossedlips.xpm" | |
33 #include "pixmaps/cry.xpm" | |
34 #include "pixmaps/embarrassed.xpm" | |
35 #include "pixmaps/kiss.xpm" | |
36 #include "pixmaps/moneymouth.xpm" | |
37 #include "pixmaps/sad.xpm" | |
38 #include "pixmaps/scream.xpm" | |
39 #include "pixmaps/smile.xpm" | |
40 #include "pixmaps/smile8.xpm" | |
41 #include "pixmaps/think.xpm" | |
42 #include "pixmaps/tongue.xpm" | |
43 #include "pixmaps/wink.xpm" | |
44 #include "pixmaps/yell.xpm" | |
45 | |
46 #define DEFAULT_FONT_NAME "helvetica" | |
47 #define MAX_SIZE 7 | |
48 | |
49 gint font_sizes [] = { 80, 100, 120, 140, 200, 300, 400 }; | |
50 | |
51 #define BORDER_SIZE 3 | |
52 #define MIN_HEIGHT 20 | |
53 #define HR_HEIGHT 2 | |
54 | |
55 #define TYPE_TEXT 0 | |
56 #define TYPE_SMILEY 1 | |
57 #define TYPE_IMG 2 | |
58 #define TYPE_SEP 3 | |
59 #define TYPE_BR 4 | |
60 #define TYPE_COMMENT 5 | |
61 | |
62 typedef struct _GtkIMHtmlBit GtkIMHtmlBit; | |
63 typedef struct _FontDetail FontDetail; | |
64 | |
65 struct _GtkIMHtmlBit { | |
66 gint type; | |
67 | |
68 gchar *text; | |
69 GdkPixmap *pm; | |
70 GdkBitmap *bm; | |
71 | |
72 GdkFont *font; | |
73 GdkColor *fore; | |
74 GdkColor *back; | |
75 GdkColor *bg; | |
76 gboolean underline; | |
77 gboolean strike; | |
78 gchar *url; | |
79 | |
80 GList *chunks; | |
81 }; | |
82 | |
83 struct _FontDetail { | |
84 gushort size; | |
85 gchar *face; | |
86 GdkColor *fore; | |
87 GdkColor *back; | |
88 }; | |
89 | |
90 struct line_info { | |
91 gint x; | |
92 gint y; | |
93 gint width; | |
94 gint height; | |
95 gint ascent; | |
96 | |
97 gboolean selected; | |
98 gchar *sel_start; | |
99 gchar *sel_end; | |
100 | |
101 gchar *text; | |
102 GtkIMHtmlBit *bit; | |
103 }; | |
104 | |
105 struct url_widget { | |
106 gint x; | |
107 gint y; | |
108 gint width; | |
109 gint height; | |
110 gchar *url; | |
111 }; | |
112 | |
113 static GtkLayoutClass *parent_class = NULL; | |
114 | |
115 enum { | |
116 TARGET_STRING, | |
117 TARGET_TEXT, | |
118 TARGET_COMPOUND_TEXT | |
119 }; | |
120 | |
121 enum { | |
122 URL_CLICKED, | |
123 LAST_SIGNAL | |
124 }; | |
125 static guint signals [LAST_SIGNAL] = { 0 }; | |
126 | |
127 static void gtk_imhtml_draw_bit (GtkIMHtml *, GtkIMHtmlBit *); | |
128 static GdkColor *gtk_imhtml_get_color (const gchar *); | |
129 static gint gtk_imhtml_motion_notify_event (GtkWidget *, GdkEventMotion *); | |
130 | |
131 static void | |
132 gtk_imhtml_destroy (GtkObject *object) | |
133 { | |
134 GtkIMHtml *imhtml; | |
135 | |
136 imhtml = GTK_IMHTML (object); | |
137 | |
138 while (imhtml->bits) { | |
139 GtkIMHtmlBit *bit = imhtml->bits->data; | |
140 imhtml->bits = g_list_remove (imhtml->bits, bit); | |
141 if (bit->text) | |
142 g_free (bit->text); | |
143 if (bit->font) | |
144 gdk_font_unref (bit->font); | |
145 if (bit->fore) | |
146 gdk_color_free (bit->fore); | |
147 if (bit->back) | |
148 gdk_color_free (bit->back); | |
149 if (bit->bg) | |
150 gdk_color_free (bit->bg); | |
151 if (bit->url) | |
152 g_free (bit->url); | |
153 if (bit->pm) | |
154 gdk_pixmap_unref (bit->pm); | |
155 if (bit->bm) | |
156 gdk_bitmap_unref (bit->bm); | |
157 while (bit->chunks) { | |
158 GtkObject *obj = bit->chunks->data; | |
159 struct line_info *li = gtk_object_get_user_data (obj); | |
160 if (li->text) | |
161 g_free (li->text); | |
162 g_free (li); | |
163 bit->chunks = g_list_remove (bit->chunks, obj); | |
164 } | |
165 g_free (bit); | |
166 } | |
167 | |
168 while (imhtml->urls) { | |
169 g_free (imhtml->urls->data); | |
170 imhtml->urls = g_list_remove (imhtml->urls, imhtml->urls->data); | |
171 } | |
172 | |
173 if (imhtml->selected_text) | |
174 g_string_free (imhtml->selected_text, TRUE); | |
175 | |
176 gdk_font_unref (imhtml->default_font); | |
177 gdk_color_free (imhtml->default_fg_color); | |
178 | |
179 gdk_cursor_destroy (imhtml->hand_cursor); | |
180 gdk_cursor_destroy (imhtml->arrow_cursor); | |
181 | |
182 g_hash_table_destroy (imhtml->smiley_hash); | |
183 | |
184 if (GTK_OBJECT_CLASS (parent_class)->destroy != NULL) | |
185 (* GTK_OBJECT_CLASS (parent_class)->destroy) (object); | |
186 } | |
187 | |
188 static void | |
189 gtk_imhtml_realize (GtkWidget *widget) | |
190 { | |
191 GtkIMHtml *imhtml; | |
192 | |
193 g_return_if_fail (widget != NULL); | |
194 g_return_if_fail (GTK_IS_IMHTML (widget)); | |
195 | |
196 imhtml = GTK_IMHTML (widget); | |
197 | |
198 if (GTK_WIDGET_CLASS (parent_class)->realize) | |
199 (* GTK_WIDGET_CLASS (parent_class)->realize) (widget); | |
200 | |
201 widget->style = gtk_style_attach (widget->style, widget->window); | |
202 gdk_window_set_events (imhtml->layout.bin_window, | |
203 (gdk_window_get_events (imhtml->layout.bin_window) | |
204 | GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | |
205 | GDK_POINTER_MOTION_MASK | GDK_EXPOSURE_MASK)); | |
206 | |
207 gdk_window_set_cursor (widget->window, imhtml->arrow_cursor); | |
208 | |
209 gdk_window_set_background (GTK_LAYOUT (imhtml)->bin_window, >K_WIDGET (imhtml)->style->white); | |
210 } | |
211 | |
212 static void | |
213 gtk_imhtml_unrealize (GtkWidget *widget) | |
214 { | |
215 if (GTK_WIDGET_CLASS (parent_class)->unrealize) | |
216 (* GTK_WIDGET_CLASS (parent_class)->unrealize) (widget); | |
217 } | |
218 | |
219 static void | |
220 gtk_imhtml_draw (GtkWidget *widget, | |
221 GdkRectangle *area) | |
222 { | |
223 if (GTK_WIDGET_CLASS (parent_class)->draw) | |
224 (* GTK_WIDGET_CLASS (parent_class)->draw) (widget, area); | |
225 } | |
226 | |
227 static void | |
228 gtk_imhtml_style_set (GtkWidget *widget, | |
229 GtkStyle *style) | |
230 { | |
231 GtkIMHtml *imhtml; | |
232 | |
233 g_return_if_fail (widget != NULL); | |
234 g_return_if_fail (GTK_IS_IMHTML (widget)); | |
235 if (!GTK_WIDGET_REALIZED (widget)) | |
236 return; | |
237 | |
238 imhtml = GTK_IMHTML (widget); | |
239 | |
240 gdk_window_set_background (GTK_LAYOUT (imhtml)->bin_window, >K_WIDGET (imhtml)->style->white); | |
241 } | |
242 | |
243 static gint | |
244 gtk_imhtml_expose (GtkWidget *widget, | |
245 GdkEventExpose *event) | |
246 { | |
247 if (GTK_WIDGET_CLASS (parent_class)->expose_event) | |
248 (* GTK_WIDGET_CLASS (parent_class)->expose_event) (widget, event); | |
249 | |
250 return TRUE; | |
251 } | |
252 | |
253 static void | |
254 gtk_imhtml_redraw_all (GtkIMHtml *imhtml) | |
255 { | |
256 GList *b; | |
257 GtkIMHtmlBit *bit; | |
258 GtkAdjustment *vadj; | |
259 gfloat oldvalue; | |
260 | |
261 vadj = GTK_LAYOUT (imhtml)->vadjustment; | |
262 oldvalue = vadj->value / vadj->upper; | |
263 | |
264 b = imhtml->bits; | |
265 while (b) { | |
266 bit = b->data; | |
267 b = g_list_next (b); | |
268 while (bit->chunks) { | |
269 GtkObject *obj = bit->chunks->data; | |
270 struct line_info *li = gtk_object_get_user_data (obj); | |
271 if (li->text) | |
272 g_free (li->text); | |
273 g_free (li); | |
274 bit->chunks = g_list_remove (bit->chunks, obj); | |
275 } | |
276 } | |
277 | |
278 g_list_free (imhtml->line); | |
279 imhtml->line = NULL; | |
280 | |
281 while (imhtml->urls) { | |
282 g_free (imhtml->urls->data); | |
283 imhtml->urls = g_list_remove (imhtml->urls, imhtml->urls->data); | |
284 } | |
285 | |
286 while (GTK_LAYOUT (imhtml)->children) | |
287 gtk_container_remove (GTK_CONTAINER (imhtml), | |
288 *(GtkWidget **)GTK_LAYOUT (imhtml)->children->data); | |
289 | |
290 imhtml->x = BORDER_SIZE; | |
291 imhtml->y = BORDER_SIZE + 10; | |
292 imhtml->llheight = 0; | |
293 imhtml->llascent = 0; | |
294 | |
295 b = imhtml->bits; | |
296 while (b) { | |
297 gtk_imhtml_draw_bit (imhtml, b->data); | |
298 b = g_list_next (b); | |
299 } | |
300 | |
301 gtk_adjustment_set_value (vadj, vadj->upper * oldvalue); | |
302 } | |
303 | |
304 static void | |
305 gtk_imhtml_size_allocate (GtkWidget *widget, | |
306 GtkAllocation *allocation) | |
307 { | |
308 GtkIMHtml *imhtml; | |
309 | |
310 g_return_if_fail (widget != NULL); | |
311 g_return_if_fail (GTK_IS_IMHTML (widget)); | |
312 g_return_if_fail (allocation != NULL); | |
313 | |
314 imhtml = GTK_IMHTML (widget); | |
315 | |
316 if (GTK_WIDGET_CLASS (parent_class)->size_allocate) | |
317 ( *GTK_WIDGET_CLASS (parent_class)->size_allocate) (widget, allocation); | |
318 | |
319 if (allocation->width == imhtml->xsize) | |
320 return; | |
321 | |
322 imhtml->x = BORDER_SIZE; | |
323 imhtml->y = BORDER_SIZE + 10; | |
324 imhtml->llheight = 0; | |
325 imhtml->llascent = 0; | |
326 | |
327 imhtml->xsize = allocation->width; | |
328 | |
329 gtk_imhtml_redraw_all (imhtml); | |
330 } | |
331 | |
332 static void | |
333 gtk_imhtml_select_none (GtkIMHtml *imhtml) | |
334 { | |
335 GList *bits; | |
336 GList *chunks; | |
337 GtkIMHtmlBit *bit; | |
338 struct line_info *chunk; | |
339 GtkWidget *darea; | |
340 | |
341 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
342 | |
343 bits = imhtml->bits; | |
344 while (bits) { | |
345 bit = bits->data; | |
346 chunks = bit->chunks; | |
347 | |
348 while (chunks) { | |
349 darea = chunks->data; | |
350 chunk = gtk_object_get_user_data (GTK_OBJECT (darea)); | |
351 | |
352 if (chunk->selected) | |
353 gtk_widget_queue_draw (darea); | |
354 chunk->selected = FALSE; | |
355 chunk->sel_start = chunk->sel_end = NULL; | |
356 | |
357 chunks = g_list_next (chunks); | |
358 } | |
359 | |
360 bits = g_list_next (bits); | |
361 } | |
362 } | |
363 | |
364 static gchar* | |
365 get_position (struct line_info *chunk, | |
366 gint x, | |
367 gboolean smileys) | |
368 { | |
369 gint width = x - chunk->x; | |
370 gchar *text; | |
371 gchar *pos; | |
372 guint total = 0; | |
373 | |
374 switch (chunk->bit->type) { | |
375 case TYPE_TEXT: | |
376 case TYPE_COMMENT: | |
377 text = chunk->text; | |
378 break; | |
379 case TYPE_SMILEY: | |
380 if (smileys) | |
381 return NULL; | |
382 else | |
383 text = chunk->text; | |
384 break; | |
385 default: | |
386 return NULL; | |
387 break; | |
388 } | |
389 | |
390 if (width <= 0) | |
391 return text; | |
392 | |
393 for (pos = text; *pos != '\0'; pos++) { | |
394 gint char_width = gdk_text_width (chunk->bit->font, pos, 1); | |
395 if ((width > total) && (width <= total + char_width)) { | |
396 if (width < total + (char_width >> 1)) | |
397 return pos; | |
398 else | |
399 return ++pos; | |
400 } | |
401 total += char_width; | |
402 } | |
403 | |
404 return pos; | |
405 } | |
406 | |
407 static GString* | |
408 append_to_sel (GString *string, | |
409 struct line_info *chunk, | |
410 gboolean smileys) | |
411 { | |
412 GString *new_string; | |
413 gchar *buf; | |
414 gchar *start; | |
415 gint length; | |
416 | |
417 switch (chunk->bit->type) { | |
418 case TYPE_TEXT: | |
419 case TYPE_COMMENT: | |
420 start = (chunk->sel_start == NULL) ? chunk->text : chunk->sel_start; | |
421 length = (chunk->sel_end == NULL) ? strlen (start) : chunk->sel_end - start; | |
422 if (length <= 0) | |
423 return string; | |
424 buf = g_strndup (start, length); | |
425 break; | |
426 case TYPE_SMILEY: | |
427 if (smileys) { | |
428 start = (chunk->sel_start == NULL) ? chunk->bit->text : chunk->sel_start; | |
429 length = (chunk->sel_end == NULL) ? strlen (start) : chunk->sel_end - start; | |
430 if (length <= 0) | |
431 return string; | |
432 buf = g_strndup (start, length); | |
433 } else { | |
434 start = (chunk->sel_start == NULL) ? chunk->text : chunk->sel_start; | |
435 length = (chunk->sel_end == NULL) ? strlen (start) : chunk->sel_end - start; | |
436 if (length <= 0) | |
437 return string; | |
438 buf = g_strndup (start, length); | |
439 } | |
440 break; | |
441 case TYPE_BR: | |
442 buf = g_strdup ("\n"); | |
443 break; | |
444 default: | |
445 return string; | |
446 break; | |
447 } | |
448 | |
449 new_string = g_string_append (string, buf); | |
450 g_free (buf); | |
451 | |
452 return new_string; | |
453 } | |
454 | |
455 #define COORDS_IN_CHUNK(xx, yy) (((xx) < chunk->x + chunk->width) && \ | |
456 ((yy) < chunk->y + chunk->height)) | |
457 | |
458 static void | |
459 gtk_imhtml_select_bits (GtkIMHtml *imhtml) | |
460 { | |
461 GList *bits; | |
462 GList *chunks; | |
463 GtkIMHtmlBit *bit; | |
464 struct line_info *chunk; | |
465 GtkWidget *darea; | |
466 | |
467 guint startx = imhtml->sel_startx, | |
468 starty = imhtml->sel_starty, | |
469 endx = imhtml->sel_endx, | |
470 endy = imhtml->sel_endy; | |
471 gchar *new_pos; | |
472 gint selection = 0; | |
473 gboolean smileys = imhtml->smileys; | |
474 gboolean redraw = FALSE; | |
475 gboolean got_start = FALSE; | |
476 gboolean got_end = FALSE; | |
477 | |
478 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
479 | |
480 if (!imhtml->selection) | |
481 return; | |
482 | |
483 if (imhtml->selected_text) { | |
484 g_string_free (imhtml->selected_text, TRUE); | |
485 imhtml->selected_text = g_string_new (""); | |
486 } | |
487 | |
488 bits = imhtml->bits; | |
489 while (bits) { | |
490 bit = bits->data; | |
491 chunks = bit->chunks; | |
492 | |
493 while (chunks) { | |
494 darea = chunks->data; | |
495 chunk = gtk_object_get_user_data (GTK_OBJECT (darea)); | |
496 | |
497 switch (selection) { | |
498 case 0: | |
499 if (COORDS_IN_CHUNK (startx, starty)) { | |
500 new_pos = get_position (chunk, startx, smileys); | |
501 if ( !chunk->selected || | |
502 (chunk->sel_start != new_pos) || | |
503 (chunk->sel_end != NULL)) | |
504 redraw = TRUE; | |
505 chunk->selected = TRUE; | |
506 chunk->sel_start = new_pos; | |
507 chunk->sel_end = NULL; | |
508 selection++; | |
509 got_start = TRUE; | |
510 } | |
511 | |
512 if (COORDS_IN_CHUNK (endx, endy)) { | |
513 if (got_start) { | |
514 new_pos = get_position (chunk, endx, smileys); | |
515 if (chunk->sel_end != new_pos) | |
516 redraw = TRUE; | |
517 if (chunk->sel_start > new_pos) { | |
518 chunk->sel_end = chunk->sel_start; | |
519 chunk->sel_start = new_pos; | |
520 } else | |
521 chunk->sel_end = new_pos; | |
522 selection = 2; | |
523 got_end = TRUE; | |
524 } else { | |
525 new_pos = get_position (chunk, endx, smileys); | |
526 if ( !chunk->selected || | |
527 (chunk->sel_start != new_pos) || | |
528 (chunk->sel_end != NULL)) | |
529 redraw = TRUE; | |
530 chunk->selected = TRUE; | |
531 chunk->sel_start = new_pos; | |
532 chunk->sel_end = NULL; | |
533 selection++; | |
534 got_end = TRUE; | |
535 } | |
536 } else if (!COORDS_IN_CHUNK (startx, starty) && !got_start) { | |
537 if (chunk->selected) | |
538 redraw = TRUE; | |
539 chunk->selected = FALSE; | |
540 chunk->sel_start = chunk->text; | |
541 chunk->sel_end = NULL; | |
542 } | |
543 | |
544 break; | |
545 case 1: | |
546 if (!got_start && COORDS_IN_CHUNK (startx, starty)) { | |
547 new_pos = get_position (chunk, startx, smileys); | |
548 if ( !chunk->selected || | |
549 (chunk->sel_end != new_pos) || | |
550 (chunk->sel_start != chunk->text)) | |
551 redraw = TRUE; | |
552 chunk->selected = TRUE; | |
553 chunk->sel_start = chunk->text; | |
554 chunk->sel_end = new_pos; | |
555 selection++; | |
556 got_start = TRUE; | |
557 } else if (!got_end && COORDS_IN_CHUNK (endx, endy)) { | |
558 new_pos = get_position (chunk, endx, smileys); | |
559 if ( !chunk->selected || | |
560 (chunk->sel_end != new_pos) || | |
561 (chunk->sel_start != chunk->text)) | |
562 redraw = TRUE; | |
563 chunk->selected = TRUE; | |
564 chunk->sel_start = chunk->text; | |
565 chunk->sel_end = new_pos; | |
566 selection++; | |
567 got_end = TRUE; | |
568 } else { | |
569 if ( !chunk->selected || | |
570 (chunk->sel_end != new_pos) || | |
571 (chunk->sel_start != NULL)) | |
572 redraw = TRUE; | |
573 chunk->selected = TRUE; | |
574 chunk->sel_start = chunk->text; | |
575 chunk->sel_end = NULL; | |
576 } | |
577 | |
578 break; | |
579 case 2: | |
580 if ( chunk->selected || | |
581 (chunk->sel_start != chunk->text) || | |
582 (chunk->sel_end != NULL)) | |
583 redraw = TRUE; | |
584 chunk->selected = FALSE; | |
585 chunk->sel_start = chunk->text; | |
586 chunk->sel_end = NULL; | |
587 break; | |
588 } | |
589 | |
590 if (chunk->selected == TRUE) | |
591 imhtml->selected_text = append_to_sel (imhtml->selected_text, | |
592 chunk, smileys); | |
593 | |
594 if (redraw) { | |
595 gtk_widget_queue_draw (darea); | |
596 redraw = FALSE; | |
597 } | |
598 | |
599 chunks = g_list_next (chunks); | |
600 } | |
601 | |
602 bits = g_list_next (bits); | |
603 } | |
604 } | |
605 | |
606 static gint | |
607 scroll_timeout (GtkIMHtml *imhtml) | |
608 { | |
609 GdkEventMotion event; | |
610 gint x, y; | |
611 GdkModifierType mask; | |
612 | |
613 imhtml->scroll_timer = 0; | |
614 | |
615 gdk_window_get_pointer (imhtml->layout.bin_window, &x, &y, &mask); | |
616 | |
617 if (mask & GDK_BUTTON1_MASK) { | |
618 event.is_hint = 0; | |
619 event.x = x; | |
620 event.y = y; | |
621 event.state = mask; | |
622 | |
623 gtk_imhtml_motion_notify_event (GTK_WIDGET (imhtml), &event); | |
624 } | |
625 | |
626 return FALSE; | |
627 } | |
628 | |
629 static gint | |
630 gtk_imhtml_motion_notify_event (GtkWidget *widget, | |
631 GdkEventMotion *event) | |
632 { | |
633 gint x, y; | |
634 GdkModifierType state; | |
635 GtkIMHtml *imhtml = GTK_IMHTML (widget); | |
636 GtkAdjustment *vadj = GTK_LAYOUT (widget)->vadjustment; | |
637 GtkAdjustment *hadj = GTK_LAYOUT (widget)->hadjustment; | |
638 | |
639 if (event->is_hint) | |
640 gdk_window_get_pointer (event->window, &x, &y, &state); | |
641 else { | |
642 x = event->x + hadj->value; | |
643 y = event->y + vadj->value; | |
644 state = event->state; | |
645 } | |
646 | |
647 if (state & GDK_BUTTON1_MASK) { | |
648 gint diff; | |
649 gint height = vadj->page_size; | |
650 gint yy = y - vadj->value; | |
651 | |
652 if (((yy < 0) || (yy > height)) && | |
653 (imhtml->scroll_timer == 0) && | |
654 (vadj->upper > vadj->page_size)) { | |
655 imhtml->scroll_timer = gtk_timeout_add (100, | |
656 (GtkFunction) scroll_timeout, | |
657 imhtml); | |
658 diff = (yy < 0) ? (yy >> 1) : ((yy - height) >> 1); | |
659 gtk_adjustment_set_value (vadj, | |
660 MIN (vadj->value + diff, vadj->upper - height + 20)); | |
661 } | |
662 | |
663 if (imhtml->selection) { | |
664 imhtml->sel_endx = MAX (x, 0); | |
665 imhtml->sel_endy = MAX (y, 0); | |
666 gtk_imhtml_select_bits (imhtml); | |
667 } | |
668 } else { | |
669 GList *urls = imhtml->urls; | |
670 struct url_widget *uw; | |
671 | |
672 while (urls) { | |
673 uw = (struct url_widget *) urls->data; | |
674 if ((x > uw->x) && (x < uw->x + uw->width) && | |
675 (y > uw->y) && (y < uw->y + uw->height)) { | |
676 gdk_window_set_cursor (imhtml->layout.bin_window, imhtml->hand_cursor); | |
677 return TRUE; | |
678 } | |
679 urls = g_list_next (urls); | |
680 } | |
681 } | |
682 | |
683 gdk_window_set_cursor (imhtml->layout.bin_window, imhtml->arrow_cursor); | |
684 | |
685 return TRUE; | |
686 } | |
687 | |
688 static gint | |
689 gtk_imhtml_button_press_event (GtkWidget *widget, | |
690 GdkEventButton *event) | |
691 { | |
692 GtkIMHtml *imhtml = GTK_IMHTML (widget); | |
693 GtkAdjustment *vadj = GTK_LAYOUT (widget)->vadjustment; | |
694 GtkAdjustment *hadj = GTK_LAYOUT (widget)->hadjustment; | |
695 gint x, y; | |
696 | |
697 if (event->button == 1) { | |
698 x = event->x + hadj->value; | |
699 y = event->y + vadj->value; | |
700 | |
701 imhtml->sel_startx = x; | |
702 imhtml->sel_starty = y; | |
703 imhtml->selection = TRUE; | |
704 gtk_imhtml_select_none (imhtml); | |
705 } | |
706 | |
707 return TRUE; | |
708 } | |
709 | |
710 static gint | |
711 gtk_imhtml_button_release_event (GtkWidget *widget, | |
712 GdkEventButton *event) | |
713 { | |
714 GtkIMHtml *imhtml = GTK_IMHTML (widget); | |
715 GtkAdjustment *vadj = GTK_LAYOUT (widget)->vadjustment; | |
716 GtkAdjustment *hadj = GTK_LAYOUT (widget)->hadjustment; | |
717 gint x, y; | |
718 | |
719 if ((event->button == 1) && imhtml->selection) { | |
720 x = event->x + hadj->value; | |
721 y = event->y + vadj->value; | |
722 | |
723 if ((x == imhtml->sel_startx) && (y == imhtml->sel_starty)) { | |
724 imhtml->sel_startx = imhtml->sel_starty = 0; | |
725 imhtml->selection = FALSE; | |
726 gtk_imhtml_select_none (imhtml); | |
727 } else { | |
728 imhtml->sel_endx = MAX (x, 0); | |
729 imhtml->sel_endy = MAX (y, 0); | |
730 gtk_imhtml_select_bits (imhtml); | |
731 } | |
732 | |
733 gtk_selection_owner_set (widget, GDK_SELECTION_PRIMARY, event->time); | |
734 } | |
735 | |
736 return TRUE; | |
737 } | |
738 | |
739 static void | |
740 gtk_imhtml_selection_get (GtkWidget *widget, | |
741 GtkSelectionData *sel_data, | |
742 guint sel_info, | |
743 guint32 time) | |
744 { | |
745 GtkIMHtml *imhtml; | |
746 gchar *string; | |
747 gint length; | |
748 | |
749 g_return_if_fail (widget != NULL); | |
750 g_return_if_fail (GTK_IS_IMHTML (widget)); | |
751 g_return_if_fail (sel_data->selection == GDK_SELECTION_PRIMARY); | |
752 | |
753 imhtml = GTK_IMHTML (widget); | |
754 | |
755 g_return_if_fail (imhtml->selected_text != NULL); | |
756 g_return_if_fail (imhtml->selected_text->str != NULL); | |
757 | |
758 if (imhtml->selected_text->len <= 0) { | |
759 string = NULL; | |
760 length = 0; | |
761 } else { | |
762 string = g_strdup (imhtml->selected_text->str); | |
763 length = strlen (string); | |
764 } | |
765 | |
766 if (sel_info == TARGET_STRING) { | |
767 gtk_selection_data_set (sel_data, | |
768 GDK_SELECTION_TYPE_STRING, | |
769 8 * sizeof (gchar), | |
770 (guchar *) string, | |
771 length); | |
772 } else if ((sel_info == TARGET_TEXT) || (sel_info == TARGET_COMPOUND_TEXT)) { | |
773 guchar *text; | |
774 GdkAtom encoding; | |
775 gint format; | |
776 gint new_length; | |
777 | |
778 gdk_string_to_compound_text (string, &encoding, &format, &text, &new_length); | |
779 gtk_selection_data_set (sel_data, encoding, format, text, new_length); | |
780 gdk_free_compound_text (text); | |
781 } | |
782 | |
783 if (string) | |
784 g_free (string); | |
785 } | |
786 | |
787 static gint | |
788 gtk_imhtml_selection_clear_event (GtkWidget *widget, | |
789 GdkEventSelection *event) | |
790 { | |
791 GtkIMHtml *imhtml; | |
792 | |
793 g_return_val_if_fail (widget != NULL, FALSE); | |
794 g_return_val_if_fail (GTK_IS_IMHTML (widget), FALSE); | |
795 g_return_val_if_fail (event != NULL, FALSE); | |
796 g_return_val_if_fail (event->selection == GDK_SELECTION_PRIMARY, TRUE); | |
797 | |
798 if (!gtk_selection_clear (widget, event)) | |
799 return FALSE; | |
800 | |
801 imhtml = GTK_IMHTML (widget); | |
802 | |
803 gtk_imhtml_select_none (imhtml); | |
804 | |
805 return TRUE; | |
806 } | |
807 | |
808 static void | |
809 gtk_imhtml_set_scroll_adjustments (GtkLayout *layout, | |
810 GtkAdjustment *hadj, | |
811 GtkAdjustment *vadj) | |
812 { | |
813 if (parent_class->set_scroll_adjustments) | |
814 (* parent_class->set_scroll_adjustments) (layout, hadj, vadj); | |
815 } | |
816 | |
817 static void | |
818 gtk_imhtml_class_init (GtkIMHtmlClass *class) | |
819 { | |
820 GtkObjectClass *object_class; | |
821 GtkWidgetClass *widget_class; | |
822 GtkLayoutClass *layout_class; | |
823 | |
824 object_class = (GtkObjectClass*) class; | |
825 widget_class = (GtkWidgetClass*) class; | |
826 layout_class = (GtkLayoutClass*) class; | |
827 | |
828 parent_class = gtk_type_class (GTK_TYPE_LAYOUT); | |
829 | |
830 signals [URL_CLICKED] = | |
831 gtk_signal_new ("url_clicked", | |
832 GTK_RUN_FIRST, | |
833 object_class->type, | |
834 GTK_SIGNAL_OFFSET (GtkIMHtmlClass, url_clicked), | |
835 gtk_marshal_NONE__POINTER, | |
836 GTK_TYPE_NONE, 1, | |
837 GTK_TYPE_POINTER); | |
838 | |
839 gtk_object_class_add_signals (object_class, signals, LAST_SIGNAL); | |
840 | |
841 object_class->destroy = gtk_imhtml_destroy; | |
842 | |
843 widget_class->realize = gtk_imhtml_realize; | |
844 widget_class->unrealize = gtk_imhtml_unrealize; | |
845 widget_class->draw = gtk_imhtml_draw; | |
846 widget_class->style_set = gtk_imhtml_style_set; | |
847 widget_class->expose_event = gtk_imhtml_expose; | |
848 widget_class->size_allocate = gtk_imhtml_size_allocate; | |
849 widget_class->motion_notify_event = gtk_imhtml_motion_notify_event; | |
850 widget_class->button_press_event = gtk_imhtml_button_press_event; | |
851 widget_class->button_release_event = gtk_imhtml_button_release_event; | |
852 widget_class->selection_get = gtk_imhtml_selection_get; | |
853 widget_class->selection_clear_event = gtk_imhtml_selection_clear_event; | |
854 | |
855 layout_class->set_scroll_adjustments = gtk_imhtml_set_scroll_adjustments; | |
856 } | |
857 | |
858 static GdkFont* | |
859 gtk_imhtml_font_load (GtkIMHtml *imhtml, | |
860 gchar *name, | |
861 gboolean bold, | |
862 gboolean italics, | |
863 gint fontsize) | |
864 { | |
865 gchar buf [16 * 1024]; | |
866 GdkFont *font; | |
867 gint size = fontsize ? font_sizes [MIN (fontsize, MAX_SIZE) - 1] : 120; | |
868 | |
869 g_snprintf (buf, sizeof (buf), "-*-%s-%s-%c-*-*-*-%d-*-*-*-*-*-*", | |
870 name ? name : DEFAULT_FONT_NAME, | |
871 bold ? "bold" : "medium", | |
872 italics ? 'i' : 'r', | |
873 size); | |
874 font = gdk_font_load (buf); | |
875 if (font) | |
876 return font; | |
877 | |
878 if (italics) { | |
879 g_snprintf (buf, sizeof (buf), "-*-%s-%s-%c-*-*-*-%d-*-*-*-*-*-*", | |
880 name ? name : DEFAULT_FONT_NAME, | |
881 bold ? "bold" : "medium", | |
882 'o', | |
883 size); | |
884 font = gdk_font_load (buf); | |
885 if (font) | |
886 return font; | |
887 | |
888 if (bold) { | |
889 g_snprintf (buf, sizeof (buf), "-*-%s-%s-%c-*-*-*-%d-*-*-*-*-*-*", | |
890 name ? name : DEFAULT_FONT_NAME, | |
891 "bold", | |
892 'r', | |
893 size); | |
894 font = gdk_font_load (buf); | |
895 if (font) | |
896 return font; | |
897 } | |
898 } | |
899 | |
900 if (bold) { | |
901 g_snprintf (buf, sizeof (buf), "-*-%s-%s-%c-*-*-*-%d-*-*-*-*-*-*", | |
902 name ? name : DEFAULT_FONT_NAME, | |
903 "medium", | |
904 italics ? 'i' : 'r', | |
905 size); | |
906 font = gdk_font_load (buf); | |
907 if (font) | |
908 return font; | |
909 | |
910 if (italics) { | |
911 g_snprintf (buf, sizeof (buf), "-*-%s-%s-%c-*-*-*-%d-*-*-*-*-*-*", | |
912 name ? name : DEFAULT_FONT_NAME, | |
913 "medium", | |
914 'o', | |
915 size); | |
916 font = gdk_font_load (buf); | |
917 if (font) | |
918 return font; | |
919 } | |
920 } | |
921 | |
922 if (!bold && !italics) { | |
923 g_snprintf (buf, sizeof (buf), "-*-%s-medium-r-*-*-*-%d-*-*-*-*-*-*", | |
924 name ? name : DEFAULT_FONT_NAME, | |
925 size); | |
926 font = gdk_font_load (buf); | |
927 if (font) | |
928 return font; | |
929 } | |
930 | |
931 g_snprintf (buf, sizeof (buf), "-*-%s-medium-r-*-*-*-%d-*-*-*-*-*-*", | |
932 DEFAULT_FONT_NAME, | |
933 size); | |
934 font = gdk_font_load (buf); | |
935 if (font) | |
936 return font; | |
937 | |
938 if (imhtml->default_font) | |
939 return gdk_font_ref (imhtml->default_font); | |
940 | |
941 return NULL; | |
942 } | |
943 | |
944 static void | |
945 gtk_imhtml_init (GtkIMHtml *imhtml) | |
946 { | |
947 static const GtkTargetEntry targets [] = { | |
948 { "STRING", 0, TARGET_STRING }, | |
949 { "TEXT", 0, TARGET_TEXT }, | |
950 { "COMPOUND_TEXT", 0, TARGET_COMPOUND_TEXT } | |
951 }; | |
952 | |
953 imhtml->default_font = gtk_imhtml_font_load (imhtml, NULL, FALSE, FALSE, 0); | |
954 if (imhtml->default_font == NULL) | |
955 g_warning ("GtkIMHtml: Could not load default font!"); | |
956 imhtml->default_fg_color = gdk_color_copy (>K_WIDGET (imhtml)->style->black); | |
957 imhtml->hand_cursor = gdk_cursor_new (GDK_HAND2); | |
958 imhtml->arrow_cursor = gdk_cursor_new (GDK_LEFT_PTR); | |
959 | |
960 GTK_WIDGET_SET_FLAGS (GTK_WIDGET (imhtml), GTK_CAN_FOCUS); | |
961 gtk_selection_add_targets (GTK_WIDGET (imhtml), GDK_SELECTION_PRIMARY, targets, 3); | |
962 } | |
963 | |
964 GtkType | |
965 gtk_imhtml_get_type (void) | |
966 { | |
967 static GtkType imhtml_type = 0; | |
968 | |
969 if (!imhtml_type) { | |
970 static const GtkTypeInfo imhtml_info = { | |
971 "GtkIMHtml", | |
972 sizeof (GtkIMHtml), | |
973 sizeof (GtkIMHtmlClass), | |
974 (GtkClassInitFunc) gtk_imhtml_class_init, | |
975 (GtkObjectInitFunc) gtk_imhtml_init, | |
976 NULL, | |
977 NULL, | |
978 NULL | |
979 }; | |
980 | |
981 imhtml_type = gtk_type_unique (GTK_TYPE_LAYOUT, &imhtml_info); | |
982 } | |
983 | |
984 return imhtml_type; | |
985 } | |
986 | |
987 static void | |
988 gtk_imhtml_init_smiley_hash (GtkIMHtml *imhtml) | |
989 { | |
990 g_return_if_fail (imhtml != NULL); | |
991 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
992 | |
993 imhtml->smiley_hash = g_hash_table_new (g_str_hash, g_str_equal); | |
994 | |
995 gtk_imhtml_associate_smiley (imhtml, ":)", smile_xpm); | |
996 gtk_imhtml_associate_smiley (imhtml, ":-)", smile_xpm); | |
997 | |
998 gtk_imhtml_associate_smiley (imhtml, ":(", sad_xpm); | |
999 gtk_imhtml_associate_smiley (imhtml, ":-(", sad_xpm); | |
1000 | |
1001 gtk_imhtml_associate_smiley (imhtml, ";)", wink_xpm); | |
1002 gtk_imhtml_associate_smiley (imhtml, ";-)", wink_xpm); | |
1003 | |
1004 gtk_imhtml_associate_smiley (imhtml, ":-p", tongue_xpm); | |
1005 gtk_imhtml_associate_smiley (imhtml, ":-P", tongue_xpm); | |
1006 | |
1007 gtk_imhtml_associate_smiley (imhtml, "=-O", scream_xpm); | |
1008 gtk_imhtml_associate_smiley (imhtml, ":-*", kiss_xpm); | |
1009 gtk_imhtml_associate_smiley (imhtml, ">:o", yell_xpm); | |
1010 gtk_imhtml_associate_smiley (imhtml, "8-)", smile8_xpm); | |
1011 gtk_imhtml_associate_smiley (imhtml, ":-$", moneymouth_xpm); | |
1012 gtk_imhtml_associate_smiley (imhtml, ":-!", burp_xpm); | |
1013 gtk_imhtml_associate_smiley (imhtml, ":-[", embarrassed_xpm); | |
1014 gtk_imhtml_associate_smiley (imhtml, ":'(", cry_xpm); | |
1015 | |
1016 gtk_imhtml_associate_smiley (imhtml, ":-/", think_xpm); | |
1017 gtk_imhtml_associate_smiley (imhtml, ":-\\", think_xpm); | |
1018 | |
1019 gtk_imhtml_associate_smiley (imhtml, ":-X", crossedlips_xpm); | |
1020 gtk_imhtml_associate_smiley (imhtml, ":-D", bigsmile_xpm); | |
1021 gtk_imhtml_associate_smiley (imhtml, "O:-)", angel_xpm); | |
1022 } | |
1023 | |
1024 GtkWidget* | |
1025 gtk_imhtml_new (GtkAdjustment *hadj, | |
1026 GtkAdjustment *vadj) | |
1027 { | |
1028 GtkIMHtml *imhtml = gtk_type_new (GTK_TYPE_IMHTML); | |
1029 | |
1030 gtk_imhtml_set_adjustments (imhtml, hadj, vadj); | |
1031 | |
1032 imhtml->bits = NULL; | |
1033 imhtml->urls = NULL; | |
1034 | |
1035 imhtml->x = BORDER_SIZE; | |
1036 imhtml->y = BORDER_SIZE + 10; | |
1037 imhtml->llheight = 0; | |
1038 imhtml->llascent = 0; | |
1039 imhtml->line = NULL; | |
1040 | |
1041 imhtml->selected_text = g_string_new (""); | |
1042 imhtml->scroll_timer = 0; | |
1043 | |
1044 imhtml->img = NULL; | |
1045 | |
1046 imhtml->smileys = TRUE; | |
1047 imhtml->comments = FALSE; | |
1048 | |
1049 imhtml->smin = G_MAXINT; | |
1050 imhtml->smax = 0; | |
1051 gtk_imhtml_init_smiley_hash (imhtml); | |
1052 | |
1053 return GTK_WIDGET (imhtml); | |
1054 } | |
1055 | |
1056 void | |
1057 gtk_imhtml_set_adjustments (GtkIMHtml *imhtml, | |
1058 GtkAdjustment *hadj, | |
1059 GtkAdjustment *vadj) | |
1060 { | |
1061 gtk_layout_set_hadjustment (GTK_LAYOUT (imhtml), hadj); | |
1062 gtk_layout_set_vadjustment (GTK_LAYOUT (imhtml), vadj); | |
1063 } | |
1064 | |
1065 void | |
1066 gtk_imhtml_set_defaults (GtkIMHtml *imhtml, | |
1067 GdkFont *font, | |
1068 GdkColor *fg_color) | |
1069 { | |
1070 g_return_if_fail (imhtml != NULL); | |
1071 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
1072 | |
1073 if (font) { | |
1074 if (imhtml->default_font) | |
1075 gdk_font_unref (imhtml->default_font); | |
1076 imhtml->default_font = gdk_font_ref (font); | |
1077 } | |
1078 | |
1079 if (fg_color) { | |
1080 if (imhtml->default_fg_color) | |
1081 gdk_color_free (imhtml->default_fg_color); | |
1082 imhtml->default_fg_color = gdk_color_copy (fg_color); | |
1083 } | |
1084 } | |
1085 | |
1086 void | |
1087 gtk_imhtml_set_img_handler (GtkIMHtml *imhtml, | |
1088 GtkIMHtmlImage handler) | |
1089 { | |
1090 g_return_if_fail (imhtml != NULL); | |
1091 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
1092 | |
1093 imhtml->img = handler; | |
1094 } | |
1095 | |
1096 void | |
1097 gtk_imhtml_associate_smiley (GtkIMHtml *imhtml, | |
1098 gchar *text, | |
1099 gchar **xpm) | |
1100 { | |
1101 g_return_if_fail (imhtml != NULL); | |
1102 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
1103 g_return_if_fail (text != NULL); | |
1104 | |
1105 if (strlen (text) < imhtml->smin) | |
1106 imhtml->smin = strlen (text); | |
1107 | |
1108 if (strlen (text) > imhtml->smax) | |
1109 imhtml->smax = strlen (text); | |
1110 | |
1111 if (xpm == NULL) | |
1112 g_hash_table_remove (imhtml->smiley_hash, text); | |
1113 else | |
1114 g_hash_table_insert (imhtml->smiley_hash, text, xpm); | |
1115 } | |
1116 | |
1117 static gint | |
1118 draw_text (GtkWidget *widget, | |
1119 GdkEvent *event, | |
1120 gpointer data) | |
1121 { | |
1122 GtkIMHtmlBit *bit; | |
1123 struct line_info *line; | |
1124 GdkGC *gc; | |
1125 GdkColormap *cmap; | |
1126 | |
1127 line = gtk_object_get_user_data (GTK_OBJECT (widget)); | |
1128 bit = line->bit; | |
1129 gc = gdk_gc_new (widget->window); | |
1130 cmap = gdk_colormap_new (gdk_visual_get_best (), FALSE); | |
1131 | |
1132 if (bit->bg != NULL) { | |
1133 gdk_color_alloc (cmap, bit->bg); | |
1134 gdk_gc_set_foreground (gc, bit->bg); | |
1135 } else | |
1136 gdk_gc_copy (gc, widget->style->white_gc); | |
1137 | |
1138 gdk_draw_rectangle (widget->window, gc, TRUE, 0, 0, | |
1139 widget->allocation.width, | |
1140 widget->allocation.height); | |
1141 | |
1142 if (!line->text) { | |
1143 gdk_colormap_unref (cmap); | |
1144 gdk_gc_unref (gc); | |
1145 return TRUE; | |
1146 } | |
1147 | |
1148 if (bit->back != NULL) { | |
1149 gdk_color_alloc (cmap, bit->back); | |
1150 gdk_gc_set_foreground (gc, bit->back); | |
1151 gdk_draw_rectangle (widget->window, gc, TRUE, 0, 0, | |
1152 gdk_string_width (bit->font, line->text), | |
1153 widget->allocation.height); | |
1154 } | |
1155 | |
1156 if (line->selected) { | |
1157 gint width, x; | |
1158 gchar *start, *end; | |
1159 GdkColor col; | |
1160 | |
1161 if ((line->sel_start > line->sel_end) && (line->sel_end != NULL)) { | |
1162 start = line->sel_end; | |
1163 end = line->sel_start; | |
1164 } else { | |
1165 start = line->sel_start; | |
1166 end = line->sel_end; | |
1167 } | |
1168 | |
1169 if (start == NULL) | |
1170 x = 0; | |
1171 else | |
1172 x = gdk_text_width (bit->font, line->text, start - line->text); | |
1173 | |
1174 if (end == NULL) | |
1175 width = gdk_string_width (bit->font, line->text) - x; | |
1176 else | |
1177 width = gdk_text_width (bit->font, line->text, end - line->text) - x; | |
1178 | |
1179 col.red = col.green = col.blue = 0xc000; | |
1180 gdk_color_alloc (cmap, &col); | |
1181 gdk_gc_set_foreground (gc, &col); | |
1182 | |
1183 gdk_draw_rectangle (widget->window, gc, TRUE, x, 0, | |
1184 width, widget->allocation.height); | |
1185 } | |
1186 | |
1187 if (bit->url) { | |
1188 GdkColor *tc = gtk_imhtml_get_color ("#0000a0"); | |
1189 gdk_color_alloc (cmap, tc); | |
1190 gdk_gc_set_foreground (gc, tc); | |
1191 gdk_color_free (tc); | |
1192 } else if (bit->fore) { | |
1193 gdk_color_alloc (cmap, bit->fore); | |
1194 gdk_gc_set_foreground (gc, bit->fore); | |
1195 } else | |
1196 gdk_gc_copy (gc, widget->style->black_gc); | |
1197 | |
1198 gdk_draw_string (widget->window, bit->font, gc, | |
1199 0, line->ascent, line->text); | |
1200 | |
1201 if (bit->underline || bit->url) | |
1202 gdk_draw_rectangle (widget->window, gc, TRUE, | |
1203 0, line->ascent + 1, | |
1204 gdk_string_width (bit->font, line->text), 1); | |
1205 if (bit->strike) | |
1206 gdk_draw_rectangle (widget->window, gc, TRUE, | |
1207 0, line->ascent - (bit->font->ascent >> 1), | |
1208 gdk_string_width (bit->font, line->text), 1); | |
1209 | |
1210 gdk_colormap_unref (cmap); | |
1211 gdk_gc_unref (gc); | |
1212 | |
1213 return TRUE; | |
1214 } | |
1215 | |
1216 static gint | |
1217 draw_img (GtkWidget *widget, | |
1218 GdkEvent *event, | |
1219 gpointer data) | |
1220 { | |
1221 GtkIMHtmlBit *bit; | |
1222 struct line_info *line; | |
1223 GdkGC *gc; | |
1224 GdkColormap *cmap; | |
1225 gint width, height, hoff; | |
1226 | |
1227 line = gtk_object_get_user_data (GTK_OBJECT (widget)); | |
1228 bit = line->bit; | |
1229 gdk_window_get_size (bit->pm, &width, &height); | |
1230 hoff = (widget->allocation.height - height) / 2; | |
1231 gc = gdk_gc_new (widget->window); | |
1232 cmap = gdk_colormap_new (gdk_visual_get_best (), FALSE); | |
1233 | |
1234 if (bit->bg != NULL) { | |
1235 gdk_color_alloc (cmap, bit->bg); | |
1236 gdk_gc_set_foreground (gc, bit->bg); | |
1237 } else | |
1238 gdk_gc_copy (gc, widget->style->white_gc); | |
1239 | |
1240 gdk_draw_rectangle (widget->window, gc, TRUE, 0, 0, | |
1241 widget->allocation.width, | |
1242 widget->allocation.height); | |
1243 | |
1244 if (bit->back != NULL) { | |
1245 gdk_color_alloc (cmap, bit->back); | |
1246 gdk_gc_set_foreground (gc, bit->back); | |
1247 gdk_draw_rectangle (widget->window, gc, TRUE, 0, 0, | |
1248 width, widget->allocation.height); | |
1249 } | |
1250 | |
1251 gdk_draw_pixmap (widget->window, gc, bit->pm, 0, 0, 0, hoff, -1, -1); | |
1252 | |
1253 gdk_colormap_unref (cmap); | |
1254 gdk_gc_unref (gc); | |
1255 | |
1256 return TRUE; | |
1257 } | |
1258 | |
1259 static gint | |
1260 draw_line (GtkWidget *widget, | |
1261 GdkEvent *event, | |
1262 gpointer data) | |
1263 { | |
1264 GtkIMHtmlBit *bit; | |
1265 GdkDrawable *drawable; | |
1266 GdkColormap *cmap; | |
1267 GdkGC *gc; | |
1268 guint max_width; | |
1269 guint max_height; | |
1270 | |
1271 bit = data; | |
1272 drawable = widget->window; | |
1273 cmap = gdk_colormap_new (gdk_visual_get_best (), FALSE); | |
1274 gc = gdk_gc_new (drawable); | |
1275 | |
1276 if (bit->bg != NULL) { | |
1277 gdk_color_alloc (cmap, bit->bg); | |
1278 gdk_gc_set_foreground (gc, bit->bg); | |
1279 | |
1280 gdk_draw_rectangle (widget->window, gc, TRUE, 0, 0, | |
1281 widget->allocation.width, | |
1282 widget->allocation.height); | |
1283 } | |
1284 | |
1285 gdk_gc_copy (gc, widget->style->black_gc); | |
1286 | |
1287 max_width = widget->allocation.width; | |
1288 max_height = widget->allocation.height / 2; | |
1289 | |
1290 gdk_draw_rectangle (drawable, gc, | |
1291 TRUE, | |
1292 0, max_height / 2, | |
1293 max_width, max_height); | |
1294 | |
1295 gdk_colormap_unref (cmap); | |
1296 gdk_gc_unref (gc); | |
1297 | |
1298 return TRUE; | |
1299 } | |
1300 | |
1301 static gint | |
1302 click_event_box (GtkBin *bin, | |
1303 GdkEventButton *event, | |
1304 GtkIMHtml *imhtml) | |
1305 { | |
1306 struct line_info *li = gtk_object_get_user_data (GTK_OBJECT (bin->child)); | |
1307 | |
1308 if (event->button == 1 && event->type == GDK_BUTTON_PRESS) | |
1309 gtk_signal_emit (GTK_OBJECT (imhtml), signals [URL_CLICKED], li->bit->url); | |
1310 | |
1311 return TRUE; | |
1312 } | |
1313 | |
1314 static void | |
1315 new_line (GtkIMHtml *imhtml) | |
1316 { | |
1317 GList *last = g_list_last (imhtml->line); | |
1318 GtkWidget *widget; | |
1319 struct line_info *li; | |
1320 | |
1321 if (last) { | |
1322 widget = last->data; | |
1323 li = gtk_object_get_user_data (GTK_OBJECT (widget)); | |
1324 if (li->x + li->width != imhtml->xsize - BORDER_SIZE) { | |
1325 li->width = imhtml->xsize - BORDER_SIZE - li->x; | |
1326 gtk_widget_set_usize (widget, li->width, li->height); | |
1327 } | |
1328 } | |
1329 | |
1330 last = imhtml->line; | |
1331 if (last) { | |
1332 widget = last->data; | |
1333 li = gtk_object_get_user_data (GTK_OBJECT (widget)); | |
1334 if (li->height < MIN_HEIGHT) { | |
1335 while (last) { | |
1336 gint diff; | |
1337 widget = last->data; | |
1338 li = gtk_object_get_user_data (GTK_OBJECT (widget)); | |
1339 diff = MIN_HEIGHT - li->height; | |
1340 li->height = MIN_HEIGHT; | |
1341 gtk_widget_set_usize (widget, li->width, li->height); | |
1342 li->ascent += diff >> 1; | |
1343 last = g_list_next (last); | |
1344 } | |
1345 imhtml->llheight = MIN_HEIGHT; | |
1346 } | |
1347 } | |
1348 | |
1349 g_list_free (imhtml->line); | |
1350 imhtml->line = NULL; | |
1351 | |
1352 imhtml->x = BORDER_SIZE; | |
1353 imhtml->y += imhtml->llheight; | |
1354 } | |
1355 | |
1356 static void | |
1357 backwards_update (GtkIMHtml *imhtml, | |
1358 GtkIMHtmlBit *bit, | |
1359 gint height, | |
1360 gint ascent) | |
1361 { | |
1362 gint diff; | |
1363 GList *ls = NULL; | |
1364 struct line_info *li; | |
1365 struct url_widget *uw; | |
1366 | |
1367 if (height > imhtml->llheight) { | |
1368 diff = height - imhtml->llheight; | |
1369 | |
1370 ls = imhtml->line; | |
1371 while (ls) { | |
1372 GtkWidget *data = ls->data; | |
1373 li = gtk_object_get_user_data (GTK_OBJECT (data)); | |
1374 li->height += diff; | |
1375 if (ascent) | |
1376 li->ascent = ascent; | |
1377 else | |
1378 li->ascent += diff >> 1; | |
1379 gtk_widget_set_usize (data, li->width, li->height); | |
1380 ls = g_list_next (ls); | |
1381 } | |
1382 | |
1383 ls = imhtml->urls; | |
1384 while (ls) { | |
1385 uw = ls->data; | |
1386 if (uw->y + diff > imhtml->y) | |
1387 uw->y += diff; | |
1388 ls = g_list_next (ls); | |
1389 } | |
1390 | |
1391 imhtml->llheight = height; | |
1392 if (ascent) | |
1393 imhtml->llascent = ascent; | |
1394 else | |
1395 imhtml->llascent += diff >> 1; | |
1396 } | |
1397 } | |
1398 | |
1399 static GtkTooltips *tips = NULL; | |
1400 | |
1401 static void | |
1402 add_text_renderer (GtkIMHtml *imhtml, | |
1403 GtkIMHtmlBit *bit, | |
1404 gchar *text) | |
1405 { | |
1406 GtkWidget *darea; | |
1407 GtkWidget *eventbox; | |
1408 struct line_info *li; | |
1409 struct url_widget *uw; | |
1410 gint width; | |
1411 | |
1412 if (text) | |
1413 width = gdk_string_width (bit->font, text); | |
1414 else | |
1415 width = 0; | |
1416 | |
1417 darea = gtk_drawing_area_new (); | |
1418 gtk_widget_set_usize (darea, width, imhtml->llheight); | |
1419 gtk_signal_connect (GTK_OBJECT (darea), "expose_event", GTK_SIGNAL_FUNC (draw_text), NULL); | |
1420 | |
1421 li = g_new0 (struct line_info, 1); | |
1422 li->x = imhtml->x; | |
1423 li->y = imhtml->y; | |
1424 li->width = width; | |
1425 li->height = imhtml->llheight; | |
1426 if (text) | |
1427 li->ascent = MAX (imhtml->llascent, bit->font->ascent); | |
1428 else | |
1429 li->ascent = 0; | |
1430 li->text = text; | |
1431 li->bit = bit; | |
1432 | |
1433 gtk_object_set_user_data (GTK_OBJECT (darea), li); | |
1434 | |
1435 if (bit->url) { | |
1436 eventbox = gtk_event_box_new (); | |
1437 gtk_layout_put (GTK_LAYOUT (imhtml), eventbox, imhtml->x, imhtml->y); | |
1438 gtk_signal_connect (GTK_OBJECT (eventbox), "button_press_event", | |
1439 GTK_SIGNAL_FUNC (click_event_box), imhtml); | |
1440 gtk_widget_show (eventbox); | |
1441 | |
1442 gtk_container_add (GTK_CONTAINER (eventbox), darea); | |
1443 | |
1444 uw = g_new0 (struct url_widget, 1); | |
1445 uw->x = imhtml->x; | |
1446 uw->y = imhtml->y; | |
1447 uw->width = width; | |
1448 uw->height = imhtml->llheight; | |
1449 uw->url = bit->url; | |
1450 imhtml->urls = g_list_append (imhtml->urls, uw); | |
1451 | |
1452 if (!tips) | |
1453 tips = gtk_tooltips_new (); | |
1454 gtk_tooltips_set_tip (tips, eventbox, bit->url, ""); | |
1455 } else { | |
1456 gtk_layout_put (GTK_LAYOUT (imhtml), darea, imhtml->x, imhtml->y); | |
1457 } | |
1458 | |
1459 bit->chunks = g_list_append (bit->chunks, darea); | |
1460 imhtml->line = g_list_append (imhtml->line, darea); | |
1461 gtk_widget_show (darea); | |
1462 | |
1463 if (bit->bg) { | |
1464 GdkColormap *cmap; | |
1465 | |
1466 cmap = gdk_colormap_new (gdk_visual_get_best (), FALSE); | |
1467 gdk_color_alloc (cmap, bit->bg); | |
1468 gdk_window_set_background (darea->window, bit->bg); | |
1469 gdk_colormap_unref (cmap); | |
1470 } else | |
1471 gdk_window_set_background (darea->window, &darea->style->white); | |
1472 } | |
1473 | |
1474 static void | |
1475 add_img_renderer (GtkIMHtml *imhtml, | |
1476 GtkIMHtmlBit *bit) | |
1477 { | |
1478 GtkWidget *darea; | |
1479 GtkWidget *eventbox; | |
1480 struct line_info *li; | |
1481 struct url_widget *uw; | |
1482 gint width; | |
1483 | |
1484 gdk_window_get_size (bit->pm, &width, NULL); | |
1485 | |
1486 darea = gtk_drawing_area_new (); | |
1487 gtk_widget_set_usize (darea, width, imhtml->llheight); | |
1488 gtk_signal_connect (GTK_OBJECT (darea), "expose_event", GTK_SIGNAL_FUNC (draw_img), NULL); | |
1489 | |
1490 li = g_new0 (struct line_info, 1); | |
1491 li->x = imhtml->x; | |
1492 li->y = imhtml->y; | |
1493 li->width = width; | |
1494 li->height = imhtml->llheight; | |
1495 li->ascent = 0; | |
1496 li->bit = bit; | |
1497 | |
1498 gtk_object_set_user_data (GTK_OBJECT (darea), li); | |
1499 | |
1500 if (bit->url) { | |
1501 eventbox = gtk_event_box_new (); | |
1502 gtk_layout_put (GTK_LAYOUT (imhtml), eventbox, imhtml->x, imhtml->y); | |
1503 gtk_signal_connect (GTK_OBJECT (eventbox), "button_press_event", | |
1504 GTK_SIGNAL_FUNC (click_event_box), imhtml); | |
1505 gtk_widget_show (eventbox); | |
1506 | |
1507 gtk_container_add (GTK_CONTAINER (eventbox), darea); | |
1508 | |
1509 uw = g_new0 (struct url_widget, 1); | |
1510 uw->x = imhtml->x; | |
1511 uw->y = imhtml->y; | |
1512 uw->width = width; | |
1513 uw->height = imhtml->llheight; | |
1514 uw->url = bit->url; | |
1515 imhtml->urls = g_list_append (imhtml->urls, uw); | |
1516 | |
1517 if (!tips) | |
1518 tips = gtk_tooltips_new (); | |
1519 gtk_tooltips_set_tip (tips, eventbox, bit->url, ""); | |
1520 } else { | |
1521 gtk_layout_put (GTK_LAYOUT (imhtml), darea, imhtml->x, imhtml->y); | |
1522 } | |
1523 | |
1524 bit->chunks = g_list_append (bit->chunks, darea); | |
1525 imhtml->line = g_list_append (imhtml->line, darea); | |
1526 gtk_widget_show (darea); | |
1527 | |
1528 if (bit->bg) { | |
1529 GdkColormap *cmap; | |
1530 | |
1531 cmap = gdk_colormap_new (gdk_visual_get_best (), FALSE); | |
1532 gdk_color_alloc (cmap, bit->bg); | |
1533 gdk_window_set_background (darea->window, bit->bg); | |
1534 gdk_colormap_unref (cmap); | |
1535 } else | |
1536 gdk_window_set_background (darea->window, &darea->style->white); | |
1537 | |
1538 imhtml->x += width; | |
1539 } | |
1540 | |
1541 static void | |
1542 gtk_imhtml_draw_bit (GtkIMHtml *imhtml, | |
1543 GtkIMHtmlBit *bit) | |
1544 { | |
1545 gint width, height; | |
1546 GdkWindow *window; | |
1547 GdkGC *gc; | |
1548 | |
1549 g_return_if_fail (imhtml != NULL); | |
1550 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
1551 g_return_if_fail (bit != NULL); | |
1552 | |
1553 window = GTK_LAYOUT (imhtml)->bin_window; | |
1554 gc = gdk_gc_new (window); | |
1555 | |
1556 if ( (bit->type == TYPE_TEXT) || | |
1557 ((bit->type == TYPE_SMILEY) && !imhtml->smileys) || | |
1558 ((bit->type == TYPE_COMMENT) && imhtml->comments)) { | |
1559 gchar *copy = g_strdup (bit->text); | |
1560 gint pos = 0; | |
1561 gboolean seenspace = FALSE; | |
1562 gchar *tmp; | |
1563 | |
1564 height = bit->font->ascent + bit->font->descent; | |
1565 width = gdk_string_width (bit->font, bit->text); | |
1566 | |
1567 if ((imhtml->x != BORDER_SIZE) && | |
1568 ((imhtml->x + width + BORDER_SIZE + BORDER_SIZE + 5) > imhtml->xsize)) { | |
1569 gint remain = imhtml->xsize - imhtml->x - BORDER_SIZE - BORDER_SIZE - 5; | |
1570 while (gdk_text_width (bit->font, copy, pos) < remain) { | |
1571 if (copy [pos] == ' ') | |
1572 seenspace = TRUE; | |
1573 pos++; | |
1574 } | |
1575 if (seenspace) { | |
1576 while (copy [pos - 1] != ' ') pos--; | |
1577 | |
1578 tmp = g_strndup (copy, pos); | |
1579 | |
1580 backwards_update (imhtml, bit, height, bit->font->ascent); | |
1581 add_text_renderer (imhtml, bit, tmp); | |
1582 } else | |
1583 pos = 0; | |
1584 seenspace = FALSE; | |
1585 new_line (imhtml); | |
1586 imhtml->llheight = 0; | |
1587 imhtml->llascent = 0; | |
1588 } | |
1589 | |
1590 backwards_update (imhtml, bit, height, bit->font->ascent); | |
1591 | |
1592 while (pos < strlen (bit->text)) { | |
1593 width = gdk_string_width (bit->font, copy + pos); | |
1594 if (imhtml->x + width + BORDER_SIZE + BORDER_SIZE + 5 > imhtml->xsize) { | |
1595 gint newpos = 0; | |
1596 gint remain = imhtml->xsize - imhtml->x - BORDER_SIZE - BORDER_SIZE - 5; | |
1597 while (gdk_text_width (bit->font, copy + pos, newpos) < remain) { | |
1598 if (copy [pos + newpos] == ' ') | |
1599 seenspace = TRUE; | |
1600 newpos++; | |
1601 } | |
1602 | |
1603 if (seenspace) | |
1604 while (copy [pos + newpos - 1] != ' ') newpos--; | |
1605 | |
1606 if (newpos == 0) | |
1607 break; | |
1608 | |
1609 tmp = g_strndup (copy + pos, newpos); | |
1610 pos += newpos; | |
1611 | |
1612 add_text_renderer (imhtml, bit, tmp); | |
1613 | |
1614 seenspace = FALSE; | |
1615 new_line (imhtml); | |
1616 } else { | |
1617 tmp = g_strdup (copy + pos); | |
1618 | |
1619 add_text_renderer (imhtml, bit, tmp); | |
1620 | |
1621 pos = strlen (bit->text); | |
1622 | |
1623 imhtml->x += width; | |
1624 } | |
1625 } | |
1626 | |
1627 g_free (copy); | |
1628 } else if ((bit->type == TYPE_SMILEY) || (bit->type == TYPE_IMG)) { | |
1629 gdk_window_get_size (bit->pm, &width, &height); | |
1630 | |
1631 if ((imhtml->x != BORDER_SIZE) && | |
1632 ((imhtml->x + width + BORDER_SIZE + BORDER_SIZE + 5) > imhtml->xsize)) { | |
1633 new_line (imhtml); | |
1634 imhtml->llheight = 0; | |
1635 imhtml->llascent = 0; | |
1636 } else | |
1637 backwards_update (imhtml, bit, height, height * 3 / 4); | |
1638 | |
1639 add_img_renderer (imhtml, bit); | |
1640 } else if (bit->type == TYPE_BR) { | |
1641 new_line (imhtml); | |
1642 imhtml->llheight = 0; | |
1643 imhtml->llascent = 0; | |
1644 add_text_renderer (imhtml, bit, NULL); | |
1645 } else if (bit->type == TYPE_SEP) { | |
1646 GtkWidget *darea; | |
1647 if (imhtml->llheight) { | |
1648 new_line (imhtml); | |
1649 imhtml->llheight = 0; | |
1650 imhtml->llascent = 0; | |
1651 } | |
1652 darea = gtk_drawing_area_new (); | |
1653 gtk_widget_set_usize (darea, imhtml->xsize - (BORDER_SIZE * 2), HR_HEIGHT * 2); | |
1654 gtk_layout_put (GTK_LAYOUT (imhtml), darea, imhtml->x, imhtml->y); | |
1655 gtk_signal_connect (GTK_OBJECT (darea), "expose_event", | |
1656 GTK_SIGNAL_FUNC (draw_line), bit); | |
1657 gtk_widget_show (darea); | |
1658 imhtml->llheight = HR_HEIGHT * 2; | |
1659 new_line (imhtml); | |
1660 imhtml->llheight = 0; | |
1661 imhtml->llascent = 0; | |
1662 add_text_renderer (imhtml, bit, NULL); | |
1663 } | |
1664 | |
1665 gtk_layout_set_size (GTK_LAYOUT (imhtml), imhtml->xsize, imhtml->y + 5); | |
1666 | |
1667 gdk_gc_destroy (gc); | |
1668 } | |
1669 | |
1670 void | |
1671 gtk_imhtml_show_smileys (GtkIMHtml *imhtml, | |
1672 gboolean show) | |
1673 { | |
1674 g_return_if_fail (imhtml != NULL); | |
1675 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
1676 | |
1677 imhtml->smileys = show; | |
1678 | |
1679 if (GTK_WIDGET_VISIBLE (GTK_WIDGET (imhtml))) | |
1680 gtk_imhtml_redraw_all (imhtml); | |
1681 } | |
1682 | |
1683 void | |
1684 gtk_imhtml_show_comments (GtkIMHtml *imhtml, | |
1685 gboolean show) | |
1686 { | |
1687 g_return_if_fail (imhtml != NULL); | |
1688 g_return_if_fail (GTK_IS_IMHTML (imhtml)); | |
1689 | |
1690 imhtml->comments = show; | |
1691 | |
1692 if (GTK_WIDGET_VISIBLE (GTK_WIDGET (imhtml))) | |
1693 gtk_imhtml_redraw_all (imhtml); | |
1694 } | |
1695 | |
1696 static GdkColor * | |
1697 gtk_imhtml_get_color (const gchar *color) | |
1698 { | |
1699 GdkColor c; | |
1700 gboolean valid = TRUE; | |
1701 | |
1702 g_return_val_if_fail (color != NULL, NULL); | |
1703 | |
1704 c.red = 0; c.green = 0; c.blue = 0; | |
1705 | |
1706 if (!g_strcasecmp (color, "aliceblue")) { | |
1707 c.red = 0xf000; c.green = 0xf800; c.blue = 0xff00; | |
1708 } else if (!g_strcasecmp (color, "antiquewhite")) { | |
1709 c.red = 0xfa00; c.green = 0xeb00; c.blue = 0xd700; | |
1710 } else if (!g_strcasecmp (color, "aqua")) { | |
1711 c.red = 0; c.green = 0xff00; c.blue = 0xff00; | |
1712 } else if (!g_strcasecmp (color, "aquamarine")) { | |
1713 c.red = 0; c.green = 0xff00; c.blue = 0xff00; | |
1714 } else if (!g_strcasecmp (color, "azure")) { | |
1715 c.red = 0xf000; c.green = 0xff00; c.blue = 0xff00; | |
1716 } else if (!g_strcasecmp (color, "beige")) { | |
1717 c.red = 0xf500; c.green = 0xf500; c.blue = 0xdc00; | |
1718 } else if (!g_strcasecmp (color, "bisque")) { | |
1719 c.red = 0xff00; c.green = 0xe400; c.blue = 0xc400; | |
1720 } else if (!g_strcasecmp (color, "black")) { | |
1721 c.red = 0; c.green = 0; c.blue = 0; | |
1722 } else if (!g_strcasecmp (color, "blanchedalmond")) { | |
1723 c.red = 0xff00; c.green = 0xeb00; c.blue = 0xcd00; | |
1724 } else if (!g_strcasecmp (color, "blue")) { | |
1725 c.red = 0; c.green = 0; c.blue = 0xff00; | |
1726 } else if (!g_strcasecmp (color, "blueviolet")) { | |
1727 c.red = 0; c.green = 0; c.blue = 0xff00; | |
1728 } else if (!g_strcasecmp (color, "brown")) { | |
1729 c.red = 0xa500; c.green = 0x2a00; c.blue = 0x2a00; | |
1730 } else if (!g_strcasecmp (color, "burlywood")) { | |
1731 c.red = 0xde00; c.green = 0xb800; c.blue = 0x8700; | |
1732 } else if (!g_strcasecmp (color, "cadetblue")) { | |
1733 c.red = 0x5f00; c.green = 0x9e00; c.blue = 0xa000; | |
1734 } else if (!g_strcasecmp (color, "chartreuse")) { | |
1735 c.red = 0x7f00; c.green = 0xff00; c.blue = 0; | |
1736 } else if (!g_strcasecmp (color, "chocolate")) { | |
1737 c.red = 0xd200; c.green = 0x6900; c.blue = 0x1e00; | |
1738 } else if (!g_strcasecmp (color, "coral")) { | |
1739 c.red = 0xff00; c.green = 0x7f00; c.blue = 0x5000; | |
1740 } else if (!g_strcasecmp (color, "cornflowerblue")) { | |
1741 c.red = 0x6400; c.green = 0x9500; c.blue = 0xed00; | |
1742 } else if (!g_strcasecmp (color, "cornsilk")) { | |
1743 c.red = 0xff00; c.green = 0xf800; c.blue = 0xdc00; | |
1744 } else if (!g_strcasecmp (color, "crimson")) { | |
1745 c.red = 0xdc00; c.green = 0x1400; c.blue = 0x3c00; | |
1746 } else if (!g_strcasecmp (color, "cyan")) { | |
1747 c.red = 0; c.green = 0xff00; c.blue = 0xff00; | |
1748 } else if (!g_strcasecmp (color, "darkblue")) { | |
1749 c.red = 0; c.green = 0; c.blue = 0x8b00; | |
1750 } else if (!g_strcasecmp (color, "darkcyan")) { | |
1751 c.red = 0; c.green = 0x8b00; c.blue = 0x8b00; | |
1752 } else if (!g_strcasecmp (color, "darkgoldenrod")) { | |
1753 c.red = 0xb800; c.green = 0x8600; c.blue = 0x0b00; | |
1754 } else if (!g_strcasecmp (color, "darkgray")) { | |
1755 c.red = 0xa900; c.green = 0xa900; c.blue = 0xa900; | |
1756 } else if (!g_strcasecmp (color, "darkgreen")) { | |
1757 c.red = 0; c.green = 0x6400; c.blue = 0; | |
1758 } else if (!g_strcasecmp (color, "darkkhaki")) { | |
1759 c.red = 0xbd00; c.green = 0xb700; c.blue = 0x6b00; | |
1760 } else if (!g_strcasecmp (color, "darkmagenta")) { | |
1761 c.red = 0x8b00; c.green = 0; c.blue = 0x8b00; | |
1762 } else if (!g_strcasecmp (color, "darkolivegreen")) { | |
1763 c.red = 0x5500; c.green = 0x6b00; c.blue = 0x2f00; | |
1764 } else if (!g_strcasecmp (color, "darkorange")) { | |
1765 c.red = 0xff00; c.green = 0x8c00; c.blue = 0; | |
1766 } else if (!g_strcasecmp (color, "darkorchid")) { | |
1767 c.red = 0x9900; c.green = 0x3200; c.blue = 0xcc00; | |
1768 } else if (!g_strcasecmp (color, "darkred")) { | |
1769 c.red = 0x8b00; c.green = 0; c.blue = 0; | |
1770 } else if (!g_strcasecmp (color, "darksalmon")) { | |
1771 c.red = 0xe900; c.green = 0x9600; c.blue = 0x7a00; | |
1772 } else if (!g_strcasecmp (color, "darkseagreen")) { | |
1773 c.red = 0x8f00; c.green = 0xbc00; c.blue = 0x8f00; | |
1774 } else if (!g_strcasecmp (color, "darkslateblue")) { | |
1775 c.red = 0x4800; c.green = 0x3d00; c.blue = 0x8b00; | |
1776 } else if (!g_strcasecmp (color, "darkslategray")) { | |
1777 c.red = 0x2f00; c.green = 0x4f00; c.blue = 0x4f00; | |
1778 } else if (!g_strcasecmp (color, "darkturquoise")) { | |
1779 c.red = 0; c.green = 0xce00; c.blue = 0xd100; | |
1780 } else if (!g_strcasecmp (color, "darkviolet")) { | |
1781 c.red = 0x9400; c.green = 0; c.blue = 0xd300; | |
1782 } else if (!g_strcasecmp (color, "deeppink")) { | |
1783 c.red = 0xff00; c.green = 0x1400; c.blue = 0x9300; | |
1784 } else if (!g_strcasecmp (color, "deepskyblue")) { | |
1785 c.red = 0; c.green = 0xbf00; c.blue = 0xff00; | |
1786 } else if (!g_strcasecmp (color, "dimgray")) { | |
1787 c.red = 0x6900; c.green = 0x6900; c.blue = 0x6900; | |
1788 } else if (!g_strcasecmp (color, "dodgerblue")) { | |
1789 c.red = 0x1e00; c.green = 0x9000; c.blue = 0xff00; | |
1790 } else if (!g_strcasecmp (color, "firebrick")) { | |
1791 c.red = 0xb200; c.green = 0x2200; c.blue = 0x2200; | |
1792 } else if (!g_strcasecmp (color, "floralwhite")) { | |
1793 c.red = 0xff00; c.green = 0xfa00; c.blue = 0xf000; | |
1794 } else if (!g_strcasecmp (color, "forestgreen")) { | |
1795 c.red = 0x2200; c.green = 0x8b00; c.blue = 0x2200; | |
1796 } else if (!g_strcasecmp (color, "fuchsia")) { | |
1797 c.red = 0xff00; c.green = 0; c.blue = 0xff00; | |
1798 } else if (!g_strcasecmp (color, "gainsboro")) { | |
1799 c.red = 0xdc00; c.green = 0xdc00; c.blue = 0xdc00; | |
1800 } else if (!g_strcasecmp (color, "ghostwhite")) { | |
1801 c.red = 0xf800; c.green = 0xf800; c.blue = 0xff00; | |
1802 } else if (!g_strcasecmp (color, "gold")) { | |
1803 c.red = 0xff00; c.green = 0xd700; c.blue = 0; | |
1804 } else if (!g_strcasecmp (color, "goldenrod")) { | |
1805 c.red = 0xff00; c.green = 0xd700; c.blue = 0; | |
1806 } else if (!g_strcasecmp (color, "gray")) { | |
1807 c.red = 0x8000; c.green = 0x8000; c.blue = 0x8000; | |
1808 } else if (!g_strcasecmp (color, "green")) { | |
1809 c.red = 0; c.green = 0x8000; c.blue = 0; | |
1810 } else if (!g_strcasecmp (color, "greenyellow")) { | |
1811 c.red = 0; c.green = 0x8000; c.blue = 0; | |
1812 } else if (!g_strcasecmp (color, "honeydew")) { | |
1813 c.red = 0xf000; c.green = 0xff00; c.blue = 0xf000; | |
1814 } else if (!g_strcasecmp (color, "hotpink")) { | |
1815 c.red = 0xff00; c.green = 0x6900; c.blue = 0xb400; | |
1816 } else if (!g_strcasecmp (color, "indianred")) { | |
1817 c.red = 0xcd00; c.green = 0x5c00; c.blue = 0x5c00; | |
1818 } else if (!g_strcasecmp (color, "indigo")) { | |
1819 c.red = 0x4b00; c.green = 0; c.blue = 0x8200; | |
1820 } else if (!g_strcasecmp (color, "ivory")) { | |
1821 c.red = 0xff00; c.green = 0xff00; c.blue = 0xf000; | |
1822 } else if (!g_strcasecmp (color, "khaki")) { | |
1823 c.red = 0xf000; c.green = 0xe600; c.blue = 0x8c00; | |
1824 } else if (!g_strcasecmp (color, "lavender")) { | |
1825 c.red = 0xe600; c.green = 0xe600; c.blue = 0xfa00; | |
1826 } else if (!g_strcasecmp (color, "lavenderblush")) { | |
1827 c.red = 0xe600; c.green = 0xe600; c.blue = 0xfa00; | |
1828 } else if (!g_strcasecmp (color, "lawngreen")) { | |
1829 c.red = 0x7c00; c.green = 0xfc00; c.blue = 0; | |
1830 } else if (!g_strcasecmp (color, "lemonchiffon")) { | |
1831 c.red = 0xff00; c.green = 0xfa00; c.blue = 0xcd00; | |
1832 } else if (!g_strcasecmp (color, "lightblue")) { | |
1833 c.red = 0xad00; c.green = 0xd800; c.blue = 0xe600; | |
1834 } else if (!g_strcasecmp (color, "lightcoral")) { | |
1835 c.red = 0xf000; c.green = 0x8000; c.blue = 0x8000; | |
1836 } else if (!g_strcasecmp (color, "lightcyan")) { | |
1837 c.red = 0xe000; c.green = 0xff00; c.blue = 0xff00; | |
1838 } else if (!g_strcasecmp (color, "lightgoldenrodyellow")) { | |
1839 c.red = 0xfa00; c.green = 0xfa00; c.blue = 0xd200; | |
1840 } else if (!g_strcasecmp (color, "lightgreen")) { | |
1841 c.red = 0x9000; c.green = 0xee00; c.blue = 0x9000; | |
1842 } else if (!g_strcasecmp (color, "lightgray")) { | |
1843 c.red = 0xd300; c.green = 0xd300; c.blue = 0xd300; | |
1844 } else if (!g_strcasecmp (color, "lightpink")) { | |
1845 c.red = 0xff00; c.green = 0xb600; c.blue = 0xc100; | |
1846 } else if (!g_strcasecmp (color, "lightsalmon")) { | |
1847 c.red = 0xff00; c.green = 0xa000; c.blue = 0x7a00; | |
1848 } else if (!g_strcasecmp (color, "lightseagreen")) { | |
1849 c.red = 0x2000; c.green = 0xb200; c.blue = 0xaa00; | |
1850 } else if (!g_strcasecmp (color, "lightskyblue")) { | |
1851 c.red = 0x8700; c.green = 0xce00; c.blue = 0xfa00; | |
1852 } else if (!g_strcasecmp (color, "lightslategray")) { | |
1853 c.red = 0x7700; c.green = 0x8800; c.blue = 0x9900; | |
1854 } else if (!g_strcasecmp (color, "lightsteelblue")) { | |
1855 c.red = 0xb000; c.green = 0xc400; c.blue = 0xde00; | |
1856 } else if (!g_strcasecmp (color, "lightyellow")) { | |
1857 c.red = 0xff00; c.green = 0xff00; c.blue = 0xe000; | |
1858 } else if (!g_strcasecmp (color, "lime")) { | |
1859 c.red = 0; c.green = 0xff00; c.blue = 0; | |
1860 } else if (!g_strcasecmp (color, "limegreen")) { | |
1861 c.red = 0; c.green = 0xff00; c.blue = 0; | |
1862 } else if (!g_strcasecmp (color, "linen")) { | |
1863 c.red = 0xfa00; c.green = 0xf000; c.blue = 0xe600; | |
1864 } else if (!g_strcasecmp (color, "magenta")) { | |
1865 c.red = 0xff00; c.green = 0; c.blue = 0xff00; | |
1866 } else if (!g_strcasecmp (color, "maroon")) { | |
1867 c.red = 0x8000; c.green = 0; c.blue = 0; | |
1868 } else if (!g_strcasecmp (color, "mediumaquamarine")) { | |
1869 c.red = 0x6600; c.green = 0xcd00; c.blue = 0xaa00; | |
1870 } else if (!g_strcasecmp (color, "mediumblue")) { | |
1871 c.red = 0; c.green = 0; c.blue = 0xcd00; | |
1872 } else if (!g_strcasecmp (color, "mediumorchid")) { | |
1873 c.red = 0xba00; c.green = 0x5500; c.blue = 0xd300; | |
1874 } else if (!g_strcasecmp (color, "mediumpurple")) { | |
1875 c.red = 0x93; c.green = 0x7000; c.blue = 0xdb00; | |
1876 } else if (!g_strcasecmp (color, "mediumseagreen")) { | |
1877 c.red = 0x3c00; c.green = 0xb300; c.blue = 0x7100; | |
1878 } else if (!g_strcasecmp (color, "mediumslateblue")) { | |
1879 c.red = 0x7b00; c.green = 0x6800; c.blue = 0xee00; | |
1880 } else if (!g_strcasecmp (color, "mediumspringgreen")) { | |
1881 c.red = 0; c.green = 0xfa00; c.blue = 0x9a00; | |
1882 } else if (!g_strcasecmp (color, "mediumturquoise")) { | |
1883 c.red = 0x4800; c.green = 0xd100; c.blue = 0xcc00; | |
1884 } else if (!g_strcasecmp (color, "mediumvioletred")) { | |
1885 c.red = 0xc700; c.green = 0x1500; c.blue = 0x8500; | |
1886 } else if (!g_strcasecmp (color, "midnightblue")) { | |
1887 c.red = 0x1900; c.green = 0x1900; c.blue = 0x7000; | |
1888 } else if (!g_strcasecmp (color, "mintcream")) { | |
1889 c.red = 0xf500; c.green = 0xff00; c.blue = 0xfa00; | |
1890 } else if (!g_strcasecmp (color, "mistyrose")) { | |
1891 c.red = 0xff00; c.green = 0xe400; c.blue = 0xe100; | |
1892 } else if (!g_strcasecmp (color, "moccasin")) { | |
1893 c.red = 0xff00; c.green = 0xe400; c.blue = 0xb500; | |
1894 } else if (!g_strcasecmp (color, "navajowhite")) { | |
1895 c.red = 0xff00; c.green = 0xde00; c.blue = 0xad00; | |
1896 } else if (!g_strcasecmp (color, "navy")) { | |
1897 c.red = 0; c.green = 0x8000; c.blue = 0; | |
1898 } else if (!g_strcasecmp (color, "oldlace")) { | |
1899 c.red = 0xfd00; c.green = 0xf500; c.blue = 0xe600; | |
1900 } else if (!g_strcasecmp (color, "olive")) { | |
1901 c.red = 0x8000; c.green = 0x8000; c.blue = 0; | |
1902 } else if (!g_strcasecmp (color, "olivedrab")) { | |
1903 c.red = 0x8000; c.green = 0x8000; c.blue = 0; | |
1904 } else if (!g_strcasecmp (color, "orange")) { | |
1905 c.red = 0xff00; c.green = 0xa500; c.blue = 0; | |
1906 } else if (!g_strcasecmp (color, "orangered")) { | |
1907 c.red = 0xff00; c.green = 0xa500; c.blue = 0; | |
1908 } else if (!g_strcasecmp (color, "orchid")) { | |
1909 c.red = 0xda00; c.green = 0x7000; c.blue = 0xd600; | |
1910 } else if (!g_strcasecmp (color, "palegoldenrod")) { | |
1911 c.red = 0xee00; c.green = 0xe800; c.blue = 0xaa00; | |
1912 } else if (!g_strcasecmp (color, "palegreen")) { | |
1913 c.red = 0x9800; c.green = 0xfb00; c.blue = 0x9800; | |
1914 } else if (!g_strcasecmp (color, "paleturquoise")) { | |
1915 c.red = 0xaf00; c.green = 0xee00; c.blue = 0xee00; | |
1916 } else if (!g_strcasecmp (color, "palevioletred")) { | |
1917 c.red = 0xdb00; c.green = 0x7000; c.blue = 0x9300; | |
1918 } else if (!g_strcasecmp (color, "papayawhip")) { | |
1919 c.red = 0xff00; c.green = 0xef00; c.blue = 0xd500; | |
1920 } else if (!g_strcasecmp (color, "peachpuff")) { | |
1921 c.red = 0xff00; c.green = 0xda00; c.blue = 0xb900; | |
1922 } else if (!g_strcasecmp (color, "peru")) { | |
1923 c.red = 0xcd00; c.green = 0x8500; c.blue = 0x3f00; | |
1924 } else if (!g_strcasecmp (color, "pink")) { | |
1925 c.red = 0xff00; c.green = 0xc000; c.blue = 0xcb00; | |
1926 } else if (!g_strcasecmp (color, "plum")) { | |
1927 c.red = 0xdd00; c.green = 0xa000; c.blue = 0xdd00; | |
1928 } else if (!g_strcasecmp (color, "powderblue")) { | |
1929 c.red = 0xb000; c.green = 0xe000; c.blue = 0xe600; | |
1930 } else if (!g_strcasecmp (color, "purple")) { | |
1931 c.red = 0x8000; c.green = 0; c.blue = 0x8000; | |
1932 } else if (!g_strcasecmp (color, "red")) { | |
1933 c.red = 0xff00; c.green = 0; c.blue = 0; | |
1934 } else if (!g_strcasecmp (color, "rosybrown")) { | |
1935 c.red = 0xbc00; c.green = 0x8f00; c.blue = 0x8f00; | |
1936 } else if (!g_strcasecmp (color, "royalblue")) { | |
1937 c.red = 0x4100; c.green = 0x6900; c.blue = 0xe100; | |
1938 } else if (!g_strcasecmp (color, "saddlebrown")) { | |
1939 c.red = 0x8b00; c.green = 0x4500; c.blue = 0x1300; | |
1940 } else if (!g_strcasecmp (color, "salmon")) { | |
1941 c.red = 0xfa00; c.green = 0x8000; c.blue = 0x7200; | |
1942 } else if (!g_strcasecmp (color, "sandybrown")) { | |
1943 c.red = 0xf400; c.green = 0xa400; c.blue = 0x6000; | |
1944 } else if (!g_strcasecmp (color, "seagreen")) { | |
1945 c.red = 0x2e00; c.green = 0x8b00; c.blue = 0x5700; | |
1946 } else if (!g_strcasecmp (color, "seashell")) { | |
1947 c.red = 0xff00; c.green = 0xf500; c.blue = 0xee00; | |
1948 } else if (!g_strcasecmp (color, "sienna")) { | |
1949 c.red = 0xa000; c.green = 0x5200; c.blue = 0x2d00; | |
1950 } else if (!g_strcasecmp (color, "silver")) { | |
1951 c.red = 0xc000; c.green = 0xc000; c.blue = 0xc000; | |
1952 } else if (!g_strcasecmp (color, "skyblue")) { | |
1953 c.red = 0x8700; c.green = 0xce00; c.blue = 0xeb00; | |
1954 } else if (!g_strcasecmp (color, "slateblue")) { | |
1955 c.red = 0x6a00; c.green = 0x5a00; c.blue = 0xcd00; | |
1956 } else if (!g_strcasecmp (color, "slategray")) { | |
1957 c.red = 0x7000; c.green = 0x8000; c.blue = 0x9000; | |
1958 } else if (!g_strcasecmp (color, "springgreen")) { | |
1959 c.red = 0; c.green = 0xff00; c.blue = 0x7f00; | |
1960 } else if (!g_strcasecmp (color, "steelblue")) { | |
1961 c.red = 0x4600; c.green = 0x8200; c.blue = 0xb400; | |
1962 } else if (!g_strcasecmp (color, "teal")) { | |
1963 c.red = 0; c.green = 0x8000; c.blue = 0x8000; | |
1964 } else if (!g_strcasecmp (color, "thistle")) { | |
1965 c.red = 0xd800; c.green = 0xbf00; c.blue = 0xd800; | |
1966 } else if (!g_strcasecmp (color, "tomato")) { | |
1967 c.red = 0xff00; c.green = 0x6300; c.blue = 0x4700; | |
1968 } else if (!g_strcasecmp (color, "turquoise")) { | |
1969 c.red = 0x4000; c.green = 0xe000; c.blue = 0xd000; | |
1970 } else if (!g_strcasecmp (color, "violet")) { | |
1971 c.red = 0xee00; c.green = 0x8200; c.blue = 0xee00; | |
1972 } else if (!g_strcasecmp (color, "wheat")) { | |
1973 c.red = 0xf500; c.green = 0xde00; c.blue = 0xb300; | |
1974 } else if (!g_strcasecmp (color, "white")) { | |
1975 c.red = 0xfe00; c.green = 0xfe00; c.blue = 0xfe00; | |
1976 } else if (!g_strcasecmp (color, "whitesmoke")) { | |
1977 c.red = 0xfe00; c.green = 0xfe00; c.blue = 0xfe00; | |
1978 } else if (!g_strcasecmp (color, "yellow")) { | |
1979 c.red = 0xff00; c.green = 0xff00; c.blue = 0; | |
1980 } else if (!g_strcasecmp (color, "yellowgreen")) { | |
1981 c.red = 0xff00; c.green = 0xff00; c.blue = 0; | |
1982 } else { | |
1983 const gchar *hex; | |
1984 guint32 value; | |
1985 | |
1986 if (color [0] == '#') | |
1987 hex = color + 1; | |
1988 else | |
1989 hex = color; | |
1990 | |
1991 if (strlen (hex) == 6) { | |
1992 gint i = 0; | |
1993 for ( ; i < 6; i++) | |
1994 if (!isxdigit ((gint) hex [i])) | |
1995 break; | |
1996 if (i == 6) { | |
1997 sscanf (hex, "%x", &value); | |
1998 c.red = (value & 0xff0000) >> 8; | |
1999 c.green = value & 0xff00; | |
2000 c.blue = (value & 0xff) << 8; | |
2001 } else { | |
2002 valid = FALSE; | |
2003 } | |
2004 } else { | |
2005 valid = FALSE; | |
2006 } | |
2007 } | |
2008 | |
2009 if (valid) | |
2010 return gdk_color_copy (&c); | |
2011 | |
2012 return NULL; | |
2013 } | |
2014 | |
2015 static gint | |
2016 gtk_imhtml_is_smiley (GtkIMHtml *imhtml, | |
2017 const gchar *text) | |
2018 { | |
2019 gchar *tmp; | |
2020 gint i; | |
2021 | |
2022 g_return_val_if_fail (imhtml != NULL, 0); | |
2023 g_return_val_if_fail (GTK_IS_IMHTML (imhtml), 0); | |
2024 g_return_val_if_fail (text != NULL, 0); | |
2025 | |
2026 tmp = g_malloc (imhtml->smax + 1); | |
2027 | |
2028 for (i = imhtml->smin; i <= imhtml->smax; i++) { | |
2029 if (strlen (text) < i) { | |
2030 g_free (tmp); | |
2031 return 0; | |
2032 } | |
2033 g_snprintf (tmp, i + 1, "%s", text); | |
2034 if (g_hash_table_lookup (imhtml->smiley_hash, tmp)) { | |
2035 g_free (tmp); | |
2036 return i; | |
2037 } | |
2038 } | |
2039 | |
2040 g_free (tmp); | |
2041 return 0; | |
2042 } | |
2043 | |
2044 static GtkIMHtmlBit * | |
2045 gtk_imhtml_new_bit (GtkIMHtml *imhtml, | |
2046 gint type, | |
2047 gchar *text, | |
2048 gint bold, | |
2049 gint italics, | |
2050 gint underline, | |
2051 gint strike, | |
2052 FontDetail *font, | |
2053 GdkColor *bg, | |
2054 gchar *url) | |
2055 { | |
2056 GtkIMHtmlBit *bit = NULL; | |
2057 | |
2058 g_return_val_if_fail (imhtml != NULL, NULL); | |
2059 g_return_val_if_fail (GTK_IS_IMHTML (imhtml), NULL); | |
2060 | |
2061 if ((type == TYPE_TEXT) && ((text == NULL) || (strlen (text) == 0))) | |
2062 return NULL; | |
2063 | |
2064 bit = g_new0 (GtkIMHtmlBit, 1); | |
2065 bit->type = type; | |
2066 | |
2067 if ((text != NULL) && (strlen (text) != 0)) | |
2068 bit->text = g_strdup (text); | |
2069 | |
2070 if ((font != NULL) || bold || italics) { | |
2071 if (font && (bold || italics || font->size || font->face)) { | |
2072 bit->font = gtk_imhtml_font_load (imhtml, font->face, bold, italics, font->size); | |
2073 } else if (bold || italics) { | |
2074 bit->font = gtk_imhtml_font_load (imhtml, NULL, bold, italics, 0); | |
2075 } | |
2076 | |
2077 if (font && (type != TYPE_BR)) { | |
2078 if (font->fore != NULL) | |
2079 bit->fore = gdk_color_copy (font->fore); | |
2080 | |
2081 if (font->back != NULL) | |
2082 bit->back = gdk_color_copy (font->back); | |
2083 } | |
2084 } | |
2085 | |
2086 if (((bit->type == TYPE_TEXT) || (bit->type == TYPE_SMILEY) || (bit->type == TYPE_COMMENT)) && | |
2087 (bit->font == NULL)) | |
2088 bit->font = gdk_font_ref (imhtml->default_font); | |
2089 | |
2090 if (bg != NULL) | |
2091 bit->bg = gdk_color_copy (bg); | |
2092 | |
2093 bit->underline = underline; | |
2094 bit->strike = strike; | |
2095 | |
2096 if (url != NULL) | |
2097 bit->url = g_strdup (url); | |
2098 | |
2099 if (type == TYPE_SMILEY) { | |
2100 GdkColor *clr; | |
2101 | |
2102 if ((font != NULL) && (font->back != NULL)) | |
2103 clr = font->back; | |
2104 else | |
2105 clr = (bg != NULL) ? bg : >K_WIDGET (imhtml)->style->white; | |
2106 | |
2107 bit->pm = gdk_pixmap_create_from_xpm_d (GTK_WIDGET (imhtml)->window, | |
2108 &bit->bm, | |
2109 clr, | |
2110 g_hash_table_lookup (imhtml->smiley_hash, text)); | |
2111 } | |
2112 | |
2113 return bit; | |
2114 } | |
2115 | |
2116 #define NEW_TEXT_BIT gtk_imhtml_new_bit (imhtml, TYPE_TEXT, ws, bold, italics, underline, strike, \ | |
2117 fonts ? fonts->data : NULL, bg, url) | |
2118 #define NEW_SMILEY_BIT gtk_imhtml_new_bit (imhtml, TYPE_SMILEY, ws, bold, italics, underline, strike, \ | |
2119 fonts ? fonts->data : NULL, bg, url) | |
2120 #define NEW_SEP_BIT gtk_imhtml_new_bit (imhtml, TYPE_SEP, NULL, 0, 0, 0, 0, NULL, bg, NULL) | |
2121 #define NEW_BR_BIT gtk_imhtml_new_bit (imhtml, TYPE_BR, NULL, 0, 0, 0, 0, \ | |
2122 fonts ? fonts->data : NULL, bg, NULL) | |
2123 #define NEW_COMMENT_BIT gtk_imhtml_new_bit (imhtml, TYPE_COMMENT, ws, bold, italics, underline, strike, \ | |
2124 fonts ? fonts->data : NULL, bg, url) | |
2125 | |
2126 #define NEW_BIT(bit) { GtkIMHtmlBit *tmp = bit; if (tmp != NULL) \ | |
2127 newbits = g_list_append (newbits, tmp); } | |
2128 | |
2129 #define UPDATE_BG_COLORS \ | |
2130 { \ | |
2131 GdkColormap *cmap; \ | |
2132 GList *rev; \ | |
2133 cmap = gdk_colormap_new (gdk_visual_get_best (), FALSE); \ | |
2134 rev = g_list_last (newbits); \ | |
2135 while (rev) { \ | |
2136 GtkIMHtmlBit *bit = rev->data; \ | |
2137 if (bit->type == TYPE_BR) \ | |
2138 break; \ | |
2139 if (bit->bg) \ | |
2140 gdk_color_free (bit->bg); \ | |
2141 bit->bg = gdk_color_copy (bg); \ | |
2142 rev = g_list_previous (rev); \ | |
2143 } \ | |
2144 if (!rev) { \ | |
2145 rev = g_list_last (imhtml->bits); \ | |
2146 while (rev) { \ | |
2147 GList *ln; \ | |
2148 GtkIMHtmlBit *bit = rev->data; \ | |
2149 if (bit->type == TYPE_BR) \ | |
2150 break; \ | |
2151 if (bit->bg) \ | |
2152 gdk_color_free (bit->bg); \ | |
2153 bit->bg = gdk_color_copy (bg); \ | |
2154 gdk_color_alloc (cmap, bit->bg); \ | |
2155 ln = bit->chunks; \ | |
2156 while (ln) { \ | |
2157 GtkWidget *widget = ln->data; \ | |
2158 gdk_window_set_background (widget->window, bit->bg); \ | |
2159 ln = g_list_next (ln); \ | |
2160 } \ | |
2161 rev = g_list_previous (rev); \ | |
2162 } \ | |
2163 gdk_colormap_unref (cmap); \ | |
2164 } \ | |
2165 } | |
2166 | |
2167 GString* | |
2168 gtk_imhtml_append_text (GtkIMHtml *imhtml, | |
2169 const gchar *text, | |
2170 GtkIMHtmlOptions options) | |
2171 { | |
2172 const gchar *c; | |
2173 gboolean intag = FALSE; | |
2174 gboolean tagquote = FALSE; | |
2175 gboolean incomment = FALSE; | |
2176 gchar *ws; | |
2177 gchar *tag; | |
2178 gint wpos = 0; | |
2179 gint tpos = 0; | |
2180 int smilelen; | |
2181 GList *newbits = NULL; | |
2182 | |
2183 guint bold = 0, | |
2184 italics = 0, | |
2185 underline = 0, | |
2186 strike = 0, | |
2187 sub = 0, | |
2188 sup = 0, | |
2189 title = 0; | |
2190 GSList *fonts = NULL; | |
2191 GdkColor *bg = NULL; | |
2192 gchar *url = NULL; | |
2193 | |
2194 GtkAdjustment *vadj; | |
2195 gboolean scrolldown = TRUE; | |
2196 | |
2197 GString *retval = NULL; | |
2198 | |
2199 g_return_val_if_fail (imhtml != NULL, NULL); | |
2200 g_return_val_if_fail (GTK_IS_IMHTML (imhtml), NULL); | |
2201 g_return_val_if_fail (text != NULL, NULL); | |
2202 | |
2203 if (options & GTK_IMHTML_RETURN_LOG) | |
2204 retval = g_string_new (""); | |
2205 | |
2206 vadj = GTK_LAYOUT (imhtml)->vadjustment; | |
2207 if ((vadj->value < imhtml->y + 5 - GTK_WIDGET (imhtml)->allocation.height) && | |
2208 (vadj->upper >= GTK_WIDGET (imhtml)->allocation.height)) | |
2209 scrolldown = FALSE; | |
2210 | |
2211 c = text; | |
2212 ws = g_malloc (strlen (text) + 1); | |
2213 tag = g_malloc (strlen (text) + 1); | |
2214 | |
2215 ws [0] = '\0'; | |
2216 | |
2217 while (*c) { | |
2218 if (*c == '<') { | |
2219 if (intag) { | |
2220 ws [wpos] = 0; | |
2221 tag [tpos] = 0; | |
2222 tpos = 0; | |
2223 strcat (ws, tag); | |
2224 wpos = strlen (ws); | |
2225 } | |
2226 | |
2227 if (incomment) { | |
2228 ws [wpos++] = *c++; | |
2229 continue; | |
2230 } | |
2231 | |
2232 if (!g_strncasecmp (c, "<!--", strlen ("<!--"))) { | |
2233 if (!(options & GTK_IMHTML_NO_COMMENTS)) { | |
2234 ws [wpos] = 0; | |
2235 wpos = 0; | |
2236 tag [tpos] = 0; | |
2237 strcat (tag, ws); | |
2238 incomment = TRUE; | |
2239 intag = FALSE; | |
2240 } | |
2241 ws [wpos++] = *c++; | |
2242 ws [wpos++] = *c++; | |
2243 ws [wpos++] = *c++; | |
2244 ws [wpos++] = *c++; | |
2245 continue; | |
2246 } | |
2247 | |
2248 tag [tpos++] = *c++; | |
2249 intag = TRUE; | |
2250 } else if (incomment && (*c == '-') && !g_strncasecmp (c, "-->", strlen ("-->"))) { | |
2251 gchar *tmp; | |
2252 ws [wpos] = 0; | |
2253 wpos = 0; | |
2254 tmp = g_strdup (ws); | |
2255 ws [wpos] = 0; | |
2256 strcat (ws, tag); | |
2257 NEW_BIT (NEW_TEXT_BIT); | |
2258 ws [wpos] = 0; | |
2259 strcat (ws, tmp + strlen ("<!--")); | |
2260 g_free (tmp); | |
2261 NEW_BIT (NEW_COMMENT_BIT); | |
2262 incomment = FALSE; | |
2263 c += strlen ("-->"); | |
2264 } else if (*c == '>' && intag && !tagquote) { | |
2265 gboolean got_tag = FALSE; | |
2266 tag [tpos++] = *c++; | |
2267 tag [tpos] = 0; | |
2268 ws [wpos] = 0; | |
2269 | |
2270 if (!g_strcasecmp (tag, "<B>") || !g_strcasecmp (tag, "<BOLD>")) { | |
2271 got_tag = TRUE; | |
2272 NEW_BIT (NEW_TEXT_BIT); | |
2273 bold++; | |
2274 } else if (!g_strcasecmp (tag, "</B>") || !g_strcasecmp (tag, "</BOLD>")) { | |
2275 got_tag = TRUE; | |
2276 if (bold) { | |
2277 NEW_BIT (NEW_TEXT_BIT); | |
2278 bold--; | |
2279 } | |
2280 } else if (!g_strcasecmp (tag, "<I>") || !g_strcasecmp (tag, "<ITALIC>")) { | |
2281 got_tag = TRUE; | |
2282 NEW_BIT (NEW_TEXT_BIT); | |
2283 italics++; | |
2284 } else if (!g_strcasecmp (tag, "</I>") || !g_strcasecmp (tag, "</ITALIC>")) { | |
2285 got_tag = TRUE; | |
2286 if (italics) { | |
2287 NEW_BIT (NEW_TEXT_BIT); | |
2288 italics--; | |
2289 } | |
2290 } else if (!g_strcasecmp (tag, "<U>") || !g_strcasecmp (tag, "<UNDERLINE>")) { | |
2291 got_tag = TRUE; | |
2292 NEW_BIT (NEW_TEXT_BIT); | |
2293 underline++; | |
2294 } else if (!g_strcasecmp (tag, "</U>") || !g_strcasecmp (tag, "</UNDERLINE>")) { | |
2295 got_tag = TRUE; | |
2296 if (underline) { | |
2297 NEW_BIT (NEW_TEXT_BIT); | |
2298 underline--; | |
2299 } | |
2300 } else if (!g_strcasecmp (tag, "<S>") || !g_strcasecmp (tag, "<STRIKE>")) { | |
2301 got_tag = TRUE; | |
2302 NEW_BIT (NEW_TEXT_BIT); | |
2303 strike++; | |
2304 } else if (!g_strcasecmp (tag, "</S>") || !g_strcasecmp (tag, "</STRIKE>")) { | |
2305 got_tag = TRUE; | |
2306 if (strike) { | |
2307 NEW_BIT (NEW_TEXT_BIT); | |
2308 strike--; | |
2309 } | |
2310 } else if (!g_strcasecmp (tag, "<SUB>")) { | |
2311 got_tag = TRUE; | |
2312 sub++; | |
2313 } else if (!g_strcasecmp (tag, "</SUB>")) { | |
2314 got_tag = TRUE; | |
2315 if (sub) { | |
2316 sub--; | |
2317 } | |
2318 } else if (!g_strcasecmp (tag, "<SUP>")) { | |
2319 got_tag = TRUE; | |
2320 sup++; | |
2321 } else if (!g_strcasecmp (tag, "</SUP>")) { | |
2322 got_tag = TRUE; | |
2323 if (sup) { | |
2324 sup--; | |
2325 } | |
2326 } else if (!g_strcasecmp (tag, "<TITLE>")) { | |
2327 if (options & GTK_IMHTML_NO_TITLE) { | |
2328 got_tag = TRUE; | |
2329 title++; | |
2330 } else { | |
2331 intag = FALSE; | |
2332 tpos = 0; | |
2333 continue; | |
2334 } | |
2335 } else if (!g_strcasecmp (tag, "</TITLE>")) { | |
2336 if (title) { | |
2337 got_tag = TRUE; | |
2338 wpos = 0; | |
2339 ws [wpos] = '\0'; | |
2340 title--; | |
2341 } else { | |
2342 intag = FALSE; | |
2343 tpos = 0; | |
2344 continue; | |
2345 } | |
2346 } else if (!g_strcasecmp (tag, "<BR>")) { | |
2347 got_tag = TRUE; | |
2348 NEW_BIT (NEW_TEXT_BIT); | |
2349 NEW_BIT (NEW_BR_BIT); | |
2350 } else if (!g_strcasecmp (tag, "<HR>") || | |
2351 !g_strncasecmp (tag, "<HR ", strlen ("<HR "))) { | |
2352 got_tag = TRUE; | |
2353 NEW_BIT (NEW_TEXT_BIT); | |
2354 NEW_BIT (NEW_SEP_BIT); | |
2355 } else if (!g_strncasecmp (tag, "<FONT ", strlen ("<FONT "))) { | |
2356 gchar *t, *e, *a, *value; | |
2357 FontDetail *font = NULL; | |
2358 GdkColor *clr; | |
2359 gint saw; | |
2360 gint i; | |
2361 | |
2362 t = tag + strlen ("<FONT "); | |
2363 | |
2364 while (*t != '\0') { | |
2365 value = NULL; | |
2366 saw = 0; | |
2367 | |
2368 while (g_strncasecmp (t, "COLOR=", strlen ("COLOR=")) | |
2369 && g_strncasecmp (t, "BACK=", strlen ("BACK=")) | |
2370 && g_strncasecmp (t, "FACE=", strlen ("FACE=")) | |
2371 && g_strncasecmp (t, "SIZE=", strlen ("SIZE="))) { | |
2372 gboolean quote = FALSE; | |
2373 if (*t == '\0') break; | |
2374 while (*t && !((*t == ' ') && !quote)) { | |
2375 if (*t == '\"') | |
2376 quote = ! quote; | |
2377 t++; | |
2378 } | |
2379 while (*t && (*t == ' ')) t++; | |
2380 } | |
2381 | |
2382 if (!g_strncasecmp (t, "COLOR=", strlen ("COLOR="))) { | |
2383 t += strlen ("COLOR="); | |
2384 saw = 1; | |
2385 } else if (!g_strncasecmp (t, "BACK=", strlen ("BACK="))) { | |
2386 t += strlen ("BACK="); | |
2387 saw = 2; | |
2388 } else if (!g_strncasecmp (t, "FACE=", strlen ("FACE="))) { | |
2389 t += strlen ("FACE="); | |
2390 saw = 3; | |
2391 } else if (!g_strncasecmp (t, "SIZE=", strlen ("SIZE="))) { | |
2392 t += strlen ("SIZE="); | |
2393 saw = 4; | |
2394 } | |
2395 | |
2396 if (!saw) | |
2397 continue; | |
2398 | |
2399 if ((*t == '\"') || (*t == '\'')) { | |
2400 e = a = ++t; | |
2401 while (*e && (*e != *(t - 1))) e++; | |
2402 if (*e != '\0') { | |
2403 *e = '\0'; | |
2404 t = e + 1; | |
2405 value = g_strdup (a); | |
2406 } else { | |
2407 *t = '\0'; | |
2408 } | |
2409 } else { | |
2410 e = a = t; | |
2411 while (*e && !isspace ((gint) *e)) e++; | |
2412 if (*e == '\0') e--; | |
2413 *e = '\0'; | |
2414 t = e + 1; | |
2415 value = g_strdup (a); | |
2416 } | |
2417 | |
2418 if (value == NULL) | |
2419 continue; | |
2420 | |
2421 if (font == NULL) | |
2422 font = g_new0 (FontDetail, 1); | |
2423 | |
2424 switch (saw) { | |
2425 case 1: | |
2426 clr = gtk_imhtml_get_color (value); | |
2427 if (clr != NULL) { | |
2428 if ( (font->fore == NULL) && | |
2429 !(options & GTK_IMHTML_NO_COLOURS)) | |
2430 font->fore = clr; | |
2431 } | |
2432 break; | |
2433 case 2: | |
2434 clr = gtk_imhtml_get_color (value); | |
2435 if (clr != NULL) { | |
2436 if ( (font->back == NULL) && | |
2437 !(options & GTK_IMHTML_NO_COLOURS)) | |
2438 font->back = clr; | |
2439 } | |
2440 break; | |
2441 case 3: | |
2442 if ( (font->face == NULL) && | |
2443 !(options & GTK_IMHTML_NO_FONTS)) | |
2444 font->face = g_strdup (value); | |
2445 break; | |
2446 case 4: | |
2447 if ((font->size != 0) || | |
2448 (options & GTK_IMHTML_NO_SIZES)) | |
2449 break; | |
2450 | |
2451 if (isdigit ((gint) value [0])) { | |
2452 for (i = 0; i < strlen (value); i++) | |
2453 if (!isdigit ((gint) value [i])) | |
2454 break; | |
2455 if (i != strlen (value)) | |
2456 break; | |
2457 | |
2458 sscanf (value, "%hd", &font->size); | |
2459 break; | |
2460 } | |
2461 | |
2462 if ((value [0] == '+') && (value [1] != '\0')) { | |
2463 for (i = 1; i < strlen (value); i++) | |
2464 if (!isdigit ((gint) value [i])) | |
2465 break; | |
2466 if (i != strlen (value)) | |
2467 break; | |
2468 | |
2469 sscanf (value + 1, "%hd", &font->size); | |
2470 font->size += 3; | |
2471 break; | |
2472 } | |
2473 | |
2474 if ((value [0] == '-') && (value [1] != '\0')) { | |
2475 for (i = 1; i < strlen (value); i++) | |
2476 if (!isdigit ((gint) value [i])) | |
2477 break; | |
2478 if (i != strlen (value)) | |
2479 break; | |
2480 | |
2481 sscanf (value + 1, "%hd", &font->size); | |
2482 font->size = MIN (font->size, 2); | |
2483 font->size = 3 - font->size; | |
2484 break; | |
2485 } | |
2486 | |
2487 break; | |
2488 } | |
2489 | |
2490 g_free (value); | |
2491 } | |
2492 | |
2493 if (!font) { | |
2494 intag = FALSE; | |
2495 tpos = 0; | |
2496 continue; | |
2497 } | |
2498 | |
2499 if (!(font->size || font->face || font->fore || font->back)) { | |
2500 g_free (font); | |
2501 intag = FALSE; | |
2502 tpos = 0; | |
2503 continue; | |
2504 } | |
2505 | |
2506 NEW_BIT (NEW_TEXT_BIT); | |
2507 | |
2508 if (fonts) { | |
2509 FontDetail *oldfont = fonts->data; | |
2510 if (!font->size) | |
2511 font->size = oldfont->size; | |
2512 if (!font->face) | |
2513 font->face = g_strdup (oldfont->face); | |
2514 if (!font->fore && oldfont->fore) | |
2515 font->fore = gdk_color_copy (oldfont->fore); | |
2516 if (!font->back && oldfont->back) | |
2517 font->back = gdk_color_copy (oldfont->back); | |
2518 } else { | |
2519 if (!font->size) | |
2520 font->size = 3; | |
2521 if (!font->face) | |
2522 font->face = g_strdup ("helvetica"); | |
2523 } | |
2524 | |
2525 fonts = g_slist_prepend (fonts, font); | |
2526 got_tag = TRUE; | |
2527 } else if (!g_strcasecmp (tag, "</FONT>")) { | |
2528 FontDetail *font; | |
2529 | |
2530 if (fonts) { | |
2531 got_tag = TRUE; | |
2532 NEW_BIT (NEW_TEXT_BIT); | |
2533 font = fonts->data; | |
2534 fonts = g_slist_remove (fonts, font); | |
2535 g_free (font->face); | |
2536 if (font->fore) | |
2537 gdk_color_free (font->fore); | |
2538 if (font->back) | |
2539 gdk_color_free (font->back); | |
2540 g_free (font); | |
2541 } else { | |
2542 intag = FALSE; | |
2543 tpos = 0; | |
2544 continue; | |
2545 } | |
2546 } else if (!g_strncasecmp (tag, "<BODY ", strlen ("<BODY "))) { | |
2547 gchar *t, *e, *color = NULL; | |
2548 GdkColor *tmp; | |
2549 | |
2550 got_tag = TRUE; | |
2551 | |
2552 if (!(options & GTK_IMHTML_NO_COLOURS)) { | |
2553 t = tag + strlen ("<BODY"); | |
2554 do { | |
2555 gboolean quote = FALSE; | |
2556 if (*t == '\0') break; | |
2557 while (*t && !((*t == ' ') && !quote)) { | |
2558 if (*t == '\"') | |
2559 quote = ! quote; | |
2560 t++; | |
2561 } | |
2562 while (*t && (*t == ' ')) t++; | |
2563 } while (g_strncasecmp (t, "BGCOLOR=", strlen ("BGCOLOR="))); | |
2564 | |
2565 if (!g_strncasecmp (t, "BGCOLOR=", strlen ("BGCOLOR="))) { | |
2566 t += strlen ("BGCOLOR="); | |
2567 if ((*t == '\"') || (*t == '\'')) { | |
2568 e = ++t; | |
2569 while (*e && (*e != *(t - 1))) e++; | |
2570 if (*e != '\0') { | |
2571 *e = '\0'; | |
2572 color = g_strdup (t); | |
2573 } | |
2574 } else { | |
2575 e = t; | |
2576 while (*e && !isspace ((gint) *e)) e++; | |
2577 if (*e == '\0') e--; | |
2578 *e = '\0'; | |
2579 color = g_strdup (t); | |
2580 } | |
2581 | |
2582 if (color != NULL) { | |
2583 tmp = gtk_imhtml_get_color (color); | |
2584 g_free (color); | |
2585 if (tmp != NULL) { | |
2586 NEW_BIT (NEW_TEXT_BIT); | |
2587 bg = tmp; | |
2588 UPDATE_BG_COLORS; | |
2589 } | |
2590 } | |
2591 } | |
2592 } | |
2593 } else if (!g_strncasecmp (tag, "<A ", strlen ("<A "))) { | |
2594 gchar *t, *e; | |
2595 | |
2596 got_tag = TRUE; | |
2597 NEW_BIT (NEW_TEXT_BIT); | |
2598 | |
2599 if (url != NULL) | |
2600 g_free (url); | |
2601 url = NULL; | |
2602 | |
2603 t = tag + strlen ("<A"); | |
2604 do { | |
2605 gboolean quote = FALSE; | |
2606 if (*t == '\0') break; | |
2607 while (*t && !((*t == ' ') && !quote)) { | |
2608 if (*t == '\"') | |
2609 quote = ! quote; | |
2610 t++; | |
2611 } | |
2612 while (*t && (*t == ' ')) t++; | |
2613 } while (g_strncasecmp (t, "HREF=", strlen ("HREF="))); | |
2614 | |
2615 if (!g_strncasecmp (t, "HREF=", strlen ("HREF="))) { | |
2616 t += strlen ("HREF="); | |
2617 if ((*t == '\"') || (*t == '\'')) { | |
2618 e = ++t; | |
2619 while (*e && (*e != *(t - 1))) e++; | |
2620 if (*e != '\0') { | |
2621 *e = '\0'; | |
2622 url = g_strdup (t); | |
2623 } | |
2624 } else { | |
2625 e = t; | |
2626 while (*e && !isspace ((gint) *e)) e++; | |
2627 if (*e == '\0') e--; | |
2628 *e = '\0'; | |
2629 url = g_strdup (t); | |
2630 } | |
2631 } | |
2632 } else if (!g_strcasecmp (tag, "</A>")) { | |
2633 got_tag = TRUE; | |
2634 if (url != NULL) { | |
2635 NEW_BIT (NEW_TEXT_BIT); | |
2636 g_free (url); | |
2637 } | |
2638 url = NULL; | |
2639 } else if (!g_strncasecmp (tag, "<IMG ", strlen ("<IMG "))) { | |
2640 gchar *t, *e, *src = NULL; | |
2641 gchar *copy = g_strdup (tag); | |
2642 gchar **xpm; | |
2643 GdkColor *clr = NULL; | |
2644 GtkIMHtmlBit *bit; | |
2645 | |
2646 intag = FALSE; | |
2647 tpos = 0; | |
2648 | |
2649 if (imhtml->img == NULL) | |
2650 continue; | |
2651 | |
2652 t = tag + strlen ("<IMG"); | |
2653 do { | |
2654 gboolean quote = FALSE; | |
2655 if (*t == '\0') break; | |
2656 while (*t && !((*t == ' ') && !quote)) { | |
2657 if (*t == '\"') | |
2658 quote = ! quote; | |
2659 t++; | |
2660 } | |
2661 while (*t && (*t == ' ')) t++; | |
2662 } while (g_strncasecmp (t, "SRC=", strlen ("SRC="))); | |
2663 | |
2664 if (!g_strncasecmp (t, "SRC=", strlen ("SRC="))) { | |
2665 t += strlen ("SRC="); | |
2666 if ((*t == '\"') || (*t == '\'')) { | |
2667 e = ++t; | |
2668 while (*e && (*e != *(t - 1))) e++; | |
2669 if (*e != '\0') { | |
2670 *e = '\0'; | |
2671 src = g_strdup (t); | |
2672 } | |
2673 } else { | |
2674 e = t; | |
2675 while (*e && !isspace ((gint) *e)) e++; | |
2676 if (*e == '\0') e--; | |
2677 *e = '\0'; | |
2678 src = g_strdup (t); | |
2679 } | |
2680 } | |
2681 | |
2682 if (src == NULL) { | |
2683 ws [wpos] = 0; | |
2684 strcat (ws, copy); | |
2685 wpos = strlen (ws); | |
2686 g_free (copy); | |
2687 continue; | |
2688 } | |
2689 | |
2690 xpm = (* imhtml->img) (src); | |
2691 if (xpm == NULL) { | |
2692 g_free (src); | |
2693 ws [wpos] = 0; | |
2694 strcat (ws, copy); | |
2695 wpos = strlen (ws); | |
2696 g_free (copy); | |
2697 continue; | |
2698 } | |
2699 | |
2700 g_free (copy); | |
2701 | |
2702 if (!fonts || ((clr = ((FontDetail *)fonts->data)->back) == NULL)) | |
2703 clr = (bg != NULL) ? bg : >K_WIDGET (imhtml)->style->white; | |
2704 | |
2705 if (!GTK_WIDGET_REALIZED (imhtml)) | |
2706 gtk_widget_realize (GTK_WIDGET (imhtml)); | |
2707 | |
2708 bit = g_new0 (GtkIMHtmlBit, 1); | |
2709 bit->type = TYPE_IMG; | |
2710 bit->pm = gdk_pixmap_create_from_xpm_d (GTK_WIDGET (imhtml)->window, | |
2711 &bit->bm, | |
2712 clr, | |
2713 xpm); | |
2714 if (url) | |
2715 bit->url = g_strdup (url); | |
2716 | |
2717 NEW_BIT (bit); | |
2718 | |
2719 g_free (src); | |
2720 | |
2721 continue; | |
2722 } else if (!g_strcasecmp (tag, "<P>") || | |
2723 !g_strcasecmp (tag, "</P>") || | |
2724 !g_strncasecmp (tag, "<P ", strlen ("<P ")) || | |
2725 !g_strcasecmp (tag, "<PRE>") || | |
2726 !g_strcasecmp (tag, "</PRE>") || | |
2727 !g_strcasecmp (tag, "<HTML>") || | |
2728 !g_strcasecmp (tag, "</HTML>") || | |
2729 !g_strcasecmp (tag, "<BODY>") || | |
2730 !g_strcasecmp (tag, "</BODY>") || | |
2731 !g_strcasecmp (tag, "<FONT>") || | |
2732 !g_strcasecmp (tag, "<HEAD>") || | |
2733 !g_strcasecmp (tag, "</HEAD>")) { | |
2734 intag = FALSE; | |
2735 tpos = 0; | |
2736 continue; | |
2737 } | |
2738 | |
2739 if (!got_tag) { | |
2740 ws [wpos] = 0; | |
2741 strcat (ws, tag); | |
2742 wpos = strlen (ws); | |
2743 } else { | |
2744 wpos = 0; | |
2745 } | |
2746 intag = FALSE; | |
2747 tpos = 0; | |
2748 } else if (*c == '&' && !intag) { | |
2749 if (!g_strncasecmp (c, "&", 5)) { | |
2750 ws [wpos++] = '&'; | |
2751 c += 5; | |
2752 } else if (!g_strncasecmp (c, "<", 4)) { | |
2753 ws [wpos++] = '<'; | |
2754 c += 4; | |
2755 } else if (!g_strncasecmp (c, ">", 4)) { | |
2756 ws [wpos++] = '>'; | |
2757 c += 4; | |
2758 } else if (!g_strncasecmp (c, " ", 6)) { | |
2759 ws [wpos++] = ' '; | |
2760 c += 6; | |
2761 } else if (!g_strncasecmp (c, "©", 6)) { | |
2762 ws [wpos++] = '©'; | |
2763 c += 6; | |
2764 } else if (!g_strncasecmp (c, """, 6)) { | |
2765 ws [wpos++] = '\"'; | |
2766 c += 6; | |
2767 } else if (!g_strncasecmp (c, "®", 5)) { | |
2768 ws [wpos++] = '®'; | |
2769 c += 5; | |
2770 } else if (*(c + 1) == '#') { | |
2771 gint pound = 0; | |
2772 if (sscanf (c, "&#%d;", £) == 1) { | |
2773 if (*(c + 3 + (gint)log10 (pound)) != ';') { | |
2774 ws [wpos++] = *c++; | |
2775 continue; | |
2776 } | |
2777 ws [wpos++] = (gchar)pound; | |
2778 c += 2; | |
2779 while (isdigit ((gint) *c)) c++; | |
2780 if (*c == ';') c++; | |
2781 } else { | |
2782 ws [wpos++] = *c++; | |
2783 } | |
2784 } else { | |
2785 ws [wpos++] = *c++; | |
2786 } | |
2787 } else if (intag) { | |
2788 if (*c == '\"') | |
2789 tagquote = !tagquote; | |
2790 tag [tpos++] = *c++; | |
2791 } else if (incomment) { | |
2792 ws [wpos++] = *c++; | |
2793 } else if (((smilelen = gtk_imhtml_is_smiley (imhtml, c)) != 0)) { | |
2794 ws [wpos] = 0; | |
2795 wpos = 0; | |
2796 NEW_BIT (NEW_TEXT_BIT); | |
2797 g_snprintf (ws, smilelen + 1, "%s", c); | |
2798 NEW_BIT (NEW_SMILEY_BIT); | |
2799 c += smilelen; | |
2800 } else if (*c == '\n') { | |
2801 if (!(options & GTK_IMHTML_NO_NEWLINE)) { | |
2802 ws [wpos] = 0; | |
2803 wpos = 0; | |
2804 NEW_BIT (NEW_TEXT_BIT); | |
2805 NEW_BIT (NEW_BR_BIT); | |
2806 } | |
2807 c++; | |
2808 } else { | |
2809 ws [wpos++] = *c++; | |
2810 } | |
2811 } | |
2812 | |
2813 if (intag) { | |
2814 tag [tpos] = 0; | |
2815 c = tag; | |
2816 while (*c) { | |
2817 if ((smilelen = gtk_imhtml_is_smiley (imhtml, c)) != 0) { | |
2818 ws [wpos] = 0; | |
2819 wpos = 0; | |
2820 NEW_BIT (NEW_TEXT_BIT); | |
2821 g_snprintf (ws, smilelen + 1, "%s", c); | |
2822 NEW_BIT (NEW_SMILEY_BIT); | |
2823 c += smilelen; | |
2824 } else { | |
2825 ws [wpos++] = *c++; | |
2826 } | |
2827 } | |
2828 } else if (incomment) { | |
2829 ws [wpos] = 0; | |
2830 wpos = 0; | |
2831 strcat (tag, ws); | |
2832 ws [wpos] = 0; | |
2833 c = tag; | |
2834 while (*c) { | |
2835 if ((smilelen = gtk_imhtml_is_smiley (imhtml, c)) != 0) { | |
2836 ws [wpos] = 0; | |
2837 wpos = 0; | |
2838 NEW_BIT (NEW_TEXT_BIT); | |
2839 g_snprintf (ws, smilelen + 1, "%s", c); | |
2840 NEW_BIT (NEW_SMILEY_BIT); | |
2841 c += smilelen; | |
2842 } else { | |
2843 ws [wpos++] = *c++; | |
2844 } | |
2845 } | |
2846 } | |
2847 | |
2848 ws [wpos] = 0; | |
2849 NEW_BIT (NEW_TEXT_BIT); | |
2850 | |
2851 while (newbits) { | |
2852 GtkIMHtmlBit *bit = newbits->data; | |
2853 imhtml->bits = g_list_append (imhtml->bits, bit); | |
2854 newbits = g_list_remove (newbits, bit); | |
2855 gtk_imhtml_draw_bit (imhtml, bit); | |
2856 } | |
2857 | |
2858 gtk_widget_set_usize (GTK_WIDGET (imhtml), -1, imhtml->y + 5); | |
2859 | |
2860 if (!(options & GTK_IMHTML_NO_SCROLL) && | |
2861 scrolldown && | |
2862 (imhtml->y + 5 >= GTK_WIDGET (imhtml)->allocation.height)) | |
2863 gtk_adjustment_set_value (vadj, imhtml->y + 5 - GTK_WIDGET (imhtml)->allocation.height); | |
2864 | |
2865 if (url) { | |
2866 g_free (url); | |
2867 if (retval) | |
2868 retval = g_string_append (retval, "</A>"); | |
2869 } | |
2870 if (bg) | |
2871 gdk_color_free (bg); | |
2872 while (fonts) { | |
2873 FontDetail *font = fonts->data; | |
2874 fonts = g_slist_remove (fonts, font); | |
2875 g_free (font->face); | |
2876 if (font->fore) | |
2877 gdk_color_free (font->fore); | |
2878 if (font->back) | |
2879 gdk_color_free (font->back); | |
2880 g_free (font); | |
2881 if (retval) | |
2882 retval = g_string_append (retval, "</FONT>"); | |
2883 } | |
2884 if (retval) { | |
2885 while (bold) { | |
2886 retval = g_string_append (retval, "</B>"); | |
2887 bold--; | |
2888 } | |
2889 while (italics) { | |
2890 retval = g_string_append (retval, "</I>"); | |
2891 italics--; | |
2892 } | |
2893 while (underline) { | |
2894 retval = g_string_append (retval, "</U>"); | |
2895 underline--; | |
2896 } | |
2897 while (strike) { | |
2898 retval = g_string_append (retval, "</S>"); | |
2899 strike--; | |
2900 } | |
2901 while (sub) { | |
2902 retval = g_string_append (retval, "</SUB>"); | |
2903 sub--; | |
2904 } | |
2905 while (sup) { | |
2906 retval = g_string_append (retval, "</SUP>"); | |
2907 sup--; | |
2908 } | |
2909 while (title) { | |
2910 retval = g_string_append (retval, "</TITLE>"); | |
2911 title--; | |
2912 } | |
2913 } | |
2914 g_free (ws); | |
2915 g_free (tag); | |
2916 | |
2917 return retval; | |
2918 } |