diff src/gtkhtml.c @ 1:2846a03bda67

[gaim-migrate @ 10] The other missing files :) committer: Tailor Script <tailor@pidgin.im>
author Rob Flynn <gaim@robflynn.com>
date Thu, 23 Mar 2000 03:13:54 +0000
parents
children 76ab3403bf02
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/gtkhtml.c	Thu Mar 23 03:13:54 2000 +0000
@@ -0,0 +1,3574 @@
+/*
+ * gaim
+ *
+ * Copyright (C) 1998-1999, Mark Spencer <markster@marko.net>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <gtk/gtk.h>
+#include <gdk/gdkprivate.h>
+#include <gdk/gdkx.h>
+#include <gdk/gdkkeysyms.h>
+#include <X11/Xlib.h>
+#include <X11/Xatom.h>
+
+#include "gaim.h"
+#include "gtkhtml.h"
+
+#define MAX_SIZE                 7
+#define MIN_HTML_WIDTH_LINES     20
+#define MIN_HTML_HEIGHT_LINES    10
+#define BORDER_WIDTH             2
+#define SCROLL_TIME              100
+#define SCROLL_PIXELS            5
+#define KEY_SCROLL_PIXELS        10
+
+int font_sizes[] = { 80, 100, 120, 140, 200, 300, 400 };
+
+GdkFont *fixed_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *fixed_bold_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *fixed_italic_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *fixed_bold_italic_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *prop_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *prop_bold_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *prop_italic_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+GdkFont *prop_bold_italic_font[] = { NULL, NULL, NULL, NULL, NULL, NULL, NULL };
+
+struct font_state {
+        int size;
+        int owncolor;
+        int ownbg;
+        GdkColor *color;
+        GdkColor *bgcol;
+        struct font_state *next;
+};
+
+struct font_state *push_state(struct font_state *current)
+{
+        struct font_state *tmp;
+	tmp = (struct font_state *)g_new0(struct font_state, 1);
+	tmp->next = current;
+	tmp->color = current->color;
+	tmp->bgcol = current->bgcol;
+	tmp->size = current->size;
+	tmp->owncolor = 0;
+	tmp->ownbg = 0;
+	return tmp;
+}
+
+enum {
+  ARG_0,
+  ARG_HADJUSTMENT,
+  ARG_VADJUSTMENT,
+};
+
+
+enum {
+  TARGET_STRING,
+  TARGET_TEXT,
+  TARGET_COMPOUND_TEXT
+};
+
+
+static void gtk_html_class_init     (GtkHtmlClass   *klass);
+static void gtk_html_set_arg        (GtkObject      *object,
+                                     GtkArg         *arg,
+                                     guint           arg_id);
+static void gtk_html_get_arg        (GtkObject      *object,
+                                     GtkArg         *arg,
+                                     guint           arg_id);
+static void gtk_html_init           (GtkHtml        *html);
+static void gtk_html_destroy        (GtkObject      *object);
+static void gtk_html_finalize       (GtkObject      *object);
+static void gtk_html_realize        (GtkWidget      *widget);
+static void gtk_html_unrealize      (GtkWidget      *widget);
+static void gtk_html_style_set      (GtkWidget      *widget,
+                                     GtkStyle       *previous_style);
+static void gtk_html_draw_focus     (GtkWidget      *widget);
+static void gtk_html_size_request   (GtkWidget      *widget,
+                                     GtkRequisition *requisition);
+static void gtk_html_size_allocate  (GtkWidget      *widget,
+                                     GtkAllocation  *allocation);
+static void gtk_html_adjustment     (GtkAdjustment  *adjustment,
+                                     GtkHtml        *html);
+static void gtk_html_disconnect     (GtkAdjustment  *adjustment,
+                                     GtkHtml        *html);
+static void gtk_html_add_seperator  (GtkHtml        *html);
+static void gtk_html_add_pixmap     (GtkHtml        *html,
+                                     GdkPixmap *pm,
+                                     gint fit);
+static void gtk_html_add_text       (GtkHtml        *html,
+                                     GdkFont        *font,
+                                     GdkColor       *fore,
+                                     GdkColor       *back,
+                                     gchar          *chars,
+                                     gint           length,
+                                     gint           uline,
+                                     gint           strike,
+                                     gchar          *url);
+static void gtk_html_draw_bit       (GtkHtml        *html,
+                                     GtkHtmlBit     *htmlbit,
+                                     gint           redraw);
+static void gtk_html_selection_get  (GtkWidget *widget,
+                                     GtkSelectionData  *selection_data,
+                                     guint sel_info,
+                                     guint32 time);
+static gint gtk_html_selection_clear (GtkWidget *widget,
+				      GdkEventSelection *event);
+static gint gtk_html_visibility_notify (GtkWidget *widget,
+				      GdkEventVisibility *event);
+
+
+/* Event handlers */
+static void gtk_html_draw           (GtkWidget      *widget,
+                                     GdkRectangle   *area);
+static gint gtk_html_expose         (GtkWidget      *widget,
+                                     GdkEventExpose *event);
+static gint gtk_html_button_press   (GtkWidget      *widget,
+                                     GdkEventButton *event);
+static gint gtk_html_button_release (GtkWidget      *widget,
+                                     GdkEventButton *event);
+static gint gtk_html_motion_notify  (GtkWidget      *widget,
+                                     GdkEventMotion *event);
+static gint gtk_html_key_press      (GtkWidget      *widget,
+				     GdkEventKey    *event);
+static gint gtk_html_leave_notify   (GtkWidget      *widget,
+				     GdkEventCrossing *event);
+
+static gint gtk_html_tooltip_timeout(gpointer data);
+
+
+static void clear_area              (GtkHtml *html,
+                                     GdkRectangle *area);
+static void expose_html             (GtkHtml *html,
+                                     GdkRectangle *area,
+                                     gboolean cursor);
+static void scroll_down             (GtkHtml *html,
+                                     gint diff0);
+static void scroll_up               (GtkHtml *html,
+                                     gint diff0);
+
+static void adjust_adj              (GtkHtml *html,
+				     GtkAdjustment *adj);
+static void resize_html             (GtkHtml *html);
+static gint html_bit_is_onscreen    (GtkHtml *html, GtkHtmlBit *hb);
+static void draw_cursor             (GtkHtml *html);
+static void undraw_cursor           (GtkHtml *html);
+
+static GtkWidgetClass *parent_class = NULL;
+
+GtkType gtk_html_get_type(void)
+{
+        static GtkType html_type = 0;
+
+        if (!html_type) {
+                static const GtkTypeInfo html_info = {
+                        "GtkHtml",
+                        sizeof(GtkHtml),
+                        sizeof(GtkHtmlClass),
+                        (GtkClassInitFunc) gtk_html_class_init,
+                        (GtkObjectInitFunc) gtk_html_init,
+                        NULL,
+                        NULL,
+                        NULL,
+                };
+                html_type = gtk_type_unique (GTK_TYPE_WIDGET, &html_info);
+        }
+        return html_type;
+}
+
+
+static void gtk_html_class_init (GtkHtmlClass *class)
+{
+        GtkObjectClass *object_class;
+        GtkWidgetClass *widget_class;
+
+        object_class = (GtkObjectClass *) class;
+        widget_class = (GtkWidgetClass *) class;
+        parent_class = gtk_type_class (GTK_TYPE_WIDGET);
+
+
+        gtk_object_add_arg_type("GtkHtml::hadjustment",
+                                GTK_TYPE_ADJUSTMENT,
+                                GTK_ARG_READWRITE | GTK_ARG_CONSTRUCT,
+                                ARG_HADJUSTMENT);
+        
+        gtk_object_add_arg_type("GtkHtml::vadjustment",
+                                GTK_TYPE_ADJUSTMENT,
+                                GTK_ARG_READWRITE | GTK_ARG_CONSTRUCT,
+                                ARG_VADJUSTMENT);
+
+        object_class->set_arg = gtk_html_set_arg;
+        object_class->get_arg = gtk_html_get_arg;
+        object_class->destroy = gtk_html_destroy;
+        object_class->finalize = gtk_html_finalize;
+
+        widget_class->realize = gtk_html_realize;
+        widget_class->unrealize = gtk_html_unrealize;
+        widget_class->style_set = gtk_html_style_set;
+        widget_class->draw_focus = gtk_html_draw_focus;
+        widget_class->size_request = gtk_html_size_request;
+        widget_class->size_allocate = gtk_html_size_allocate;
+        widget_class->draw = gtk_html_draw;
+        widget_class->expose_event = gtk_html_expose;
+        widget_class->button_press_event = gtk_html_button_press;
+        widget_class->button_release_event = gtk_html_button_release;
+	widget_class->motion_notify_event = gtk_html_motion_notify;
+	widget_class->leave_notify_event = gtk_html_leave_notify;
+        widget_class->selection_get = gtk_html_selection_get;
+	widget_class->selection_clear_event = gtk_html_selection_clear;
+	widget_class->key_press_event = gtk_html_key_press;
+	widget_class->visibility_notify_event = gtk_html_visibility_notify;
+	
+
+        widget_class->set_scroll_adjustments_signal =
+                gtk_signal_new ("set_scroll_adjustments",
+                                GTK_RUN_LAST,
+                                object_class->type,
+                                GTK_SIGNAL_OFFSET (GtkHtmlClass, set_scroll_adjustments),
+                                gtk_marshal_NONE__POINTER_POINTER,
+                                GTK_TYPE_NONE, 2, GTK_TYPE_ADJUSTMENT, GTK_TYPE_ADJUSTMENT);
+
+
+        class->set_scroll_adjustments = gtk_html_set_adjustments;
+
+}
+
+static void gtk_html_set_arg (GtkObject *object,
+                              GtkArg *arg,
+                              guint arg_id)
+{
+        GtkHtml *html;
+
+        html = GTK_HTML (object);
+
+        switch (arg_id) {
+        case ARG_HADJUSTMENT:
+                gtk_html_set_adjustments (html,
+                                          GTK_VALUE_POINTER (*arg),
+                                          html->vadj);
+                break;
+        case ARG_VADJUSTMENT:
+                gtk_html_set_adjustments (html,
+                                          html->hadj,
+                                          GTK_VALUE_POINTER (*arg));
+                break;
+        default:
+                break;
+        }
+}
+
+static void gtk_html_get_arg (GtkObject *object,
+                              GtkArg *arg,
+                              guint arg_id)
+{
+        GtkHtml *html;
+
+        html = GTK_HTML(object);
+
+        switch (arg_id) {
+        case ARG_HADJUSTMENT:
+                GTK_VALUE_POINTER (*arg) = html->hadj;
+                break;
+        case ARG_VADJUSTMENT:
+                GTK_VALUE_POINTER (*arg) = html->vadj;
+                break;
+        default:
+                arg->type = GTK_TYPE_INVALID;
+                break;
+        }
+}
+
+static void gtk_html_init (GtkHtml *html)
+{
+        static const GtkTargetEntry targets[] = {
+                { "STRING", 0, TARGET_STRING },
+                { "TEXT",   0, TARGET_TEXT },
+                { "COMPOUND_TEXT", 0, TARGET_COMPOUND_TEXT }
+        };
+
+        static const gint n_targets = sizeof(targets) / sizeof(targets[0]);
+
+        GTK_WIDGET_SET_FLAGS(html, GTK_CAN_FOCUS);
+
+        html->html_area = NULL;
+        html->hadj = NULL;
+        html->vadj = NULL;
+	html->current_x = 0;
+	html->current_y = 0;
+        html->start_sel = html->end_sel = NULL;
+        html->start_sel_x = html->start_sel_y = -1;
+        html->num_end = html->num_start = -1;
+        
+	html->html_bits = NULL;
+	html->urls = NULL;
+	html->selected_text = NULL;
+	html->tooltip_hb = NULL;
+	html->tooltip_timer = -1;
+	html->tooltip_window = NULL;
+        html->cursor_hb = NULL;
+	html->cursor_pos = 0;
+
+	html->pm = NULL;
+
+	html->editable = 0;
+	html->transparent = 0;
+
+        html->frozen = 0;
+	
+        gtk_selection_add_targets (GTK_WIDGET (html), GDK_SELECTION_PRIMARY,
+                                   targets, n_targets);
+
+
+        
+}
+
+
+GtkWidget *gtk_html_new (GtkAdjustment *hadj,
+                         GtkAdjustment *vadj)
+{
+        GtkWidget *html;
+
+        if (hadj)
+                g_return_val_if_fail (GTK_IS_ADJUSTMENT (hadj), NULL);
+        if (vadj)
+                g_return_val_if_fail (GTK_IS_ADJUSTMENT (vadj), NULL);
+
+        html = gtk_widget_new (GTK_TYPE_HTML,
+                               "hadjustment", hadj,
+                               "vadjustment", vadj,
+                               NULL);
+
+        return html;
+}
+
+
+void gtk_html_set_editable (GtkHtml *html,
+                            gboolean is_editable)
+{
+        g_return_if_fail (html != NULL);
+        g_return_if_fail (GTK_IS_HTML (html));
+  
+
+        html->editable = (is_editable != FALSE);
+
+        if (is_editable)
+                draw_cursor(html);
+        else
+                undraw_cursor(html);
+
+}
+
+void gtk_html_set_transparent( GtkHtml *html,
+                              gboolean is_transparent)
+{
+        GdkRectangle rect;
+        gint width, height;
+        GtkWidget *widget;
+
+        g_return_if_fail (html != NULL);
+        g_return_if_fail (GTK_IS_HTML (html));
+  
+
+        widget = GTK_WIDGET(html);
+        html->transparent = (is_transparent != FALSE);
+
+        if (!GTK_WIDGET_REALIZED(widget))
+                return;
+
+        html->bg_gc = NULL;
+        gdk_window_get_size (widget->window, &width, &height);
+        rect.x = 0;
+        rect.y = 0;
+        rect.width = width;
+        rect.height = height;
+        gdk_window_clear_area (widget->window, rect.x, rect.y, rect.width, rect.height);
+
+        expose_html (html, &rect, FALSE);
+        gtk_html_draw_focus ( (GtkWidget *) html);
+}
+
+
+void gtk_html_set_adjustments (GtkHtml       *html,
+                               GtkAdjustment *hadj,
+                               GtkAdjustment *vadj)
+{
+        g_return_if_fail (html != NULL);
+        g_return_if_fail (GTK_IS_HTML (html));
+        if (hadj)
+                g_return_if_fail (GTK_IS_ADJUSTMENT (hadj));
+        else
+                hadj = GTK_ADJUSTMENT (gtk_adjustment_new (0.0, 0.0, 0.0, 0.0, 0.0, 0.0));
+        if (vadj)
+                g_return_if_fail (GTK_IS_ADJUSTMENT (vadj));
+        else
+                vadj = GTK_ADJUSTMENT (gtk_adjustment_new (0.0, 0.0, 0.0, 0.0, 0.0, 0.0));
+  
+        if (html->hadj && (html->hadj != hadj)) {
+                gtk_signal_disconnect_by_data (GTK_OBJECT (html->hadj), html);
+                gtk_object_unref (GTK_OBJECT (html->hadj));
+        }
+  
+        if (html->vadj && (html->vadj != vadj)) {
+                gtk_signal_disconnect_by_data (GTK_OBJECT (html->vadj), html);
+                gtk_object_unref (GTK_OBJECT (html->vadj));
+        }
+
+        if (html->hadj != hadj) {
+                html->hadj = hadj;
+                gtk_object_ref (GTK_OBJECT (html->hadj));
+                gtk_object_sink (GTK_OBJECT (html->hadj));
+      
+                gtk_signal_connect (GTK_OBJECT (html->hadj), "changed",
+                                    (GtkSignalFunc) gtk_html_adjustment,
+                                    html);
+                gtk_signal_connect (GTK_OBJECT (html->hadj), "value_changed",
+                                    (GtkSignalFunc) gtk_html_adjustment,
+                                    html);
+                gtk_signal_connect (GTK_OBJECT (html->hadj), "disconnect",
+                                    (GtkSignalFunc) gtk_html_disconnect,
+                                    html);
+                gtk_html_adjustment (hadj, html);
+        }
+  
+        if (html->vadj != vadj) {
+                html->vadj = vadj;
+                gtk_object_ref (GTK_OBJECT (html->vadj));
+                gtk_object_sink (GTK_OBJECT (html->vadj));
+
+                gtk_signal_connect (GTK_OBJECT (html->vadj), "changed",
+                                    (GtkSignalFunc) gtk_html_adjustment,
+                                    html);
+                gtk_signal_connect (GTK_OBJECT (html->vadj), "value_changed",
+                                    (GtkSignalFunc) gtk_html_adjustment,
+                                    html);
+                gtk_signal_connect (GTK_OBJECT (html->vadj), "disconnect",
+                                    (GtkSignalFunc) gtk_html_disconnect,
+                                    html);
+                gtk_html_adjustment (vadj, html);
+        }
+}
+
+
+
+GdkColor *get_color(int colorv, GdkColormap *map)
+{
+	GdkColor *color;
+#if 0	
+	fprintf(stdout,"color is %x\n",colorv);
+#endif									
+	color = (GdkColor *)g_new0(GdkColor, 1);
+	color->red = ((colorv & 0xff0000) >> 16) * 256;
+	color->green = ((colorv & 0xff00) >> 8) * 256;
+	color->blue = ((colorv & 0xff)) * 256;
+#if 0
+	fprintf(stdout,"Colors are %d, %d, %d\n",color->red, color->green, color->blue);
+#endif
+	gdk_color_alloc(map, color);
+	return color;
+}
+
+
+
+GdkFont *font_load(char *fmt, int size)
+{
+	char buf[256];
+	g_snprintf(buf, sizeof(buf), fmt, font_sizes[size]);
+	sprintf(debug_buff,"loading font %s\n",buf);
+	debug_print(debug_buff);
+	return gdk_font_load(buf);
+}
+
+
+GdkFont *getfont(int bold, int italic, int fixed, int size)
+{
+	if (size > MAX_SIZE) size = MAX_SIZE;
+	if (size < 1) size=1;
+	size--;
+	if (fixed) {
+		if (bold) {
+			if (italic) {
+				if (!fixed_bold_italic_font[size])
+					fixed_bold_italic_font[size] = font_load(FIXED_BOLD_ITALIC_FONT,size);
+				return fixed_bold_italic_font[size];
+					
+			} else {
+				if (!fixed_bold_font[size])
+					fixed_bold_font[size] = font_load (FIXED_BOLD_FONT,size);
+				return fixed_bold_font[size];
+			}
+		} else if (italic) {
+			if (!fixed_italic_font[size])
+				fixed_italic_font[size] = font_load(FIXED_ITALIC_FONT,size);
+				
+			return fixed_italic_font[size];
+		} else {
+			if (!fixed_font[size]) 
+				fixed_font[size] = font_load(FIXED_FONT,size);
+			return fixed_font[size];
+		}
+	} else {
+		if (bold) {
+			if (italic) {
+				if (!prop_bold_italic_font[size])
+					prop_bold_italic_font[size] = font_load(PROP_BOLD_ITALIC_FONT,size);
+				return prop_bold_italic_font[size];
+					
+			} else {
+				if (!prop_bold_font[size])
+					prop_bold_font[size] = font_load (PROP_BOLD_FONT,size);
+				return prop_bold_font[size];
+			}
+		} else if (italic) {
+			if (!prop_italic_font[size])
+				prop_italic_font[size] = font_load(PROP_ITALIC_FONT,size);
+				
+			return prop_italic_font[size];
+		} else {
+			if (!prop_font[size]) 
+				prop_font[size] = font_load(PROP_FONT,size);
+			return prop_font[size];
+		}
+	}
+}
+
+
+
+
+/* 'Borrowed' from ETerm */
+GdkWindow *get_desktop_window(GtkWidget *widget)
+{
+	GdkAtom prop, type, prop2;
+        int format;
+        gint length;
+        guchar *data;
+	GtkWidget *w;
+
+        prop = gdk_atom_intern("_XROOTPMAP_ID", 1);
+        prop2 = gdk_atom_intern("_XROOTCOLOR_PIXEL", 1);
+
+        if (prop == None && prop2 == None) {
+                return NULL;
+        }
+
+
+	
+	for (w = widget; w; w = w->parent) {
+
+		if (prop != None) {
+			gdk_property_get(w->window, prop, AnyPropertyType, 0L, 1L, 0,
+					 &type, &format, &length, &data);
+		} else if (prop2 != None) {
+			gdk_property_get(w->window, prop2, AnyPropertyType, 0L, 1L, 0,
+					 &type, &format, &length, &data);
+		} else {
+			continue;
+		}
+		if (type != None) {
+			return (w->window);
+		}
+	}
+
+	return NULL;
+
+}
+
+
+
+GdkPixmap *get_desktop_pixmap(GtkWidget *widget)
+{
+        GdkPixmap *p;
+        GdkAtom prop, type, prop2;
+        int format;
+        gint length;
+        guint32 id;
+        guchar *data;
+
+        prop = gdk_atom_intern("_XROOTPMAP_ID", 1);
+        prop2 = gdk_atom_intern("_XROOTCOLOR_PIXEL", 1);
+
+
+        if (prop == None && prop2 == None) {
+                return NULL;
+        }
+
+        if (prop != None) {
+                gdk_property_get(get_desktop_window(widget), prop, AnyPropertyType, 0L, 1L, 0,
+                                 &type, &format, &length, &data);
+                if (type == XA_PIXMAP) {
+			id = data[0];
+			id += data[1] << 8;
+			id += data[2] << 16;
+			id += data[3] << 24;
+                        p = gdk_pixmap_foreign_new(id);
+                        return p;
+                }
+        }
+        if (prop2 != None) {
+/*                XGetWindowProperty(Xdisplay, desktop_window, prop2, 0L, 1L, False, AnyPropertyType,
+                                   &type, &format, &length, &after, &data);*/
+/*                if (type == XA_CARDINAL) {*/
+                        /*     D_PIXMAP(("  Solid color not yet supported.\n"));*/
+/*                        return NULL;
+                }*/
+        }
+        /*        D_PIXMAP(("No suitable attribute found.\n"));*/
+        return NULL;
+}
+
+
+static void clear_focus_area (GtkHtml *html,
+                              gint area_x,
+                              gint area_y,
+                              gint area_width,
+                              gint area_height)
+{
+        GtkWidget *widget = GTK_WIDGET (html);
+	gint x, y;
+  
+        gint ythick = BORDER_WIDTH + widget->style->klass->ythickness;
+        gint xthick = BORDER_WIDTH + widget->style->klass->xthickness;
+  
+        gint width, height;
+        
+        if (html->frozen > 0)
+        	return;
+
+	if (html->transparent) {
+		if (html->pm == NULL)
+			html->pm = get_desktop_pixmap(widget);
+
+                if (html->pm == NULL)
+                        return;
+
+		if (html->bg_gc == NULL) {
+			GdkGCValues values;
+
+                        values.tile = html->pm;
+                        values.fill = GDK_TILED;
+
+                        html->bg_gc = gdk_gc_new_with_values (html->html_area, &values,
+                                                       GDK_GC_FILL | GDK_GC_TILE);
+
+                }
+
+                gdk_window_get_deskrelative_origin(widget->window, &x, &y);
+
+		gdk_draw_pixmap(widget->window, html->bg_gc, html->pm,
+				x + area_x, y + area_y, area_x, area_y, area_width,
+				area_height);
+
+		
+        } else {
+                gdk_window_get_size (widget->style->bg_pixmap[GTK_STATE_NORMAL], &width, &height);
+
+                gdk_gc_set_ts_origin (html->bg_gc,
+                                      (- html->xoffset + xthick) % width,
+                                      (- html->yoffset + ythick) % height);
+
+                gdk_draw_rectangle (widget->window, html->bg_gc, TRUE,
+                                    area_x, area_y, area_width, area_height);
+        }
+}
+
+static void gtk_html_draw_focus (GtkWidget *widget)
+{
+        GtkHtml *html;
+        gint width, height;
+        gint x, y;
+
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+  
+        html = GTK_HTML (widget);
+  
+        if (GTK_WIDGET_DRAWABLE (widget)) {
+                gint ythick = widget->style->klass->ythickness;
+                gint xthick = widget->style->klass->xthickness;
+                gint xextra = BORDER_WIDTH;
+                gint yextra = BORDER_WIDTH;
+      
+                x = 0;
+                y = 0;
+                width = widget->allocation.width;
+                height = widget->allocation.height;
+      
+                if (GTK_WIDGET_HAS_FOCUS (widget)) {
+                        x += 1;
+                        y += 1;
+                        width -=  2;
+                        height -= 2;
+                        xextra -= 1;
+                        yextra -= 1;
+
+                        gtk_paint_focus (widget->style, widget->window,
+                                         NULL, widget, "text",
+                                         0, 0,
+                                         widget->allocation.width - 1,
+                                         widget->allocation.height - 1);
+                }
+
+                gtk_paint_shadow (widget->style, widget->window,
+                                  GTK_STATE_NORMAL, GTK_SHADOW_IN,
+                                  NULL, widget, "text",
+                                  x, y, width, height);
+
+                x += xthick;
+                y += ythick;
+                width -= 2 * xthick;
+                height -= 2 * ythick;
+
+
+                if (widget->style->bg_pixmap[GTK_STATE_NORMAL] || html->transparent) {
+                        /* top rect */
+                        clear_focus_area (html, x, y, width, yextra);
+                        /* left rect */
+                        clear_focus_area (html, x, y + yextra,
+                                          xextra, y + height - 2 * yextra);
+                        /* right rect */
+                        clear_focus_area (html, x + width - xextra, y + yextra,
+                                          xextra, height - 2 * ythick);
+                        /* bottom rect */
+                        clear_focus_area (html, x, x + height - yextra, width, yextra);
+                }
+        }
+}
+
+static void gtk_html_size_request (GtkWidget      *widget,
+                                   GtkRequisition *requisition)
+{
+        gint xthickness;
+        gint ythickness;
+        gint char_height;
+        gint char_width;
+
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+        g_return_if_fail (requisition != NULL);
+  
+        xthickness = widget->style->klass->xthickness + BORDER_WIDTH;
+        ythickness = widget->style->klass->ythickness + BORDER_WIDTH;
+  
+        char_height = MIN_HTML_HEIGHT_LINES * (widget->style->font->ascent +
+                                               widget->style->font->descent);
+  
+        char_width = MIN_HTML_WIDTH_LINES * (gdk_text_width (widget->style->font,
+                                                             "ABCDEFGHIJKLMNOPQRSTUVWXYZ",
+                                                             26) / 26);
+  
+        requisition->width  = char_width  + xthickness * 2;
+        requisition->height = char_height + ythickness * 2;
+}
+
+static void gtk_html_size_allocate (GtkWidget     *widget,
+                                    GtkAllocation *allocation)
+{
+        GtkHtml *html;
+  
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+        g_return_if_fail (allocation != NULL);
+
+	html = GTK_HTML(widget);
+
+        widget->allocation = *allocation;
+        if (GTK_WIDGET_REALIZED (widget)) {
+                gdk_window_move_resize (widget->window,
+                                        allocation->x, allocation->y,
+                                        allocation->width, allocation->height);
+      
+                gdk_window_move_resize (html->html_area,
+                              widget->style->klass->xthickness + BORDER_WIDTH,
+                              widget->style->klass->ythickness + BORDER_WIDTH,
+                                        MAX (1, (gint)widget->allocation.width -
+                                             (gint)(widget->style->klass->xthickness +
+                                                    (gint)BORDER_WIDTH) * 2),
+                                        MAX (1, (gint)widget->allocation.height -
+                                             (gint)(widget->style->klass->ythickness +
+                                                    (gint)BORDER_WIDTH) * 2));
+
+                resize_html(html);
+        }
+}
+
+static void gtk_html_draw (GtkWidget    *widget,
+                           GdkRectangle *area)
+{
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+        g_return_if_fail (area != NULL);
+  
+        if (GTK_WIDGET_DRAWABLE (widget))
+        {
+                expose_html (GTK_HTML(widget), area, TRUE);
+                gtk_widget_draw_focus (widget);
+        }
+}
+
+
+static gint gtk_html_expose (GtkWidget *widget,
+                             GdkEventExpose *event)
+{
+        GtkHtml *html;
+        
+        g_return_val_if_fail (widget != NULL, FALSE);
+        g_return_val_if_fail (GTK_IS_HTML (widget), FALSE);
+        g_return_val_if_fail (event != NULL, FALSE);
+
+        html = GTK_HTML(widget);
+  
+        if (event->window == html->html_area) {
+                expose_html(html, &event->area, TRUE);
+        } else if (event->count == 0) {
+                gtk_widget_draw_focus (widget);
+        }
+  
+        return FALSE;
+
+}
+
+
+static gint gtk_html_selection_clear (GtkWidget         *widget,
+                                      GdkEventSelection *event)
+{
+        GtkHtml *html;
+        
+        g_return_val_if_fail (widget != NULL, FALSE);
+        g_return_val_if_fail (GTK_IS_HTML (widget), FALSE);
+        g_return_val_if_fail (event != NULL, FALSE);
+  
+        /* Let the selection handling code know that the selection
+         * has been changed, since we've overriden the default handler */
+        if (!gtk_selection_clear (widget, event))
+                return FALSE;
+  
+        html = GTK_HTML (widget);
+  
+        if (event->selection == GDK_SELECTION_PRIMARY) {
+                if (html->selected_text) {
+                        GList *hbits = html->html_bits;
+                        GtkHtmlBit *hb;
+                        
+			g_free(html->selected_text);
+			html->selected_text = NULL;
+			html->start_sel = NULL;
+			html->end_sel = NULL;
+			html->num_start = 0;
+			html->num_end = 0;
+			while(hbits) {
+				hb = (GtkHtmlBit *)hbits->data;
+				if (hb->was_selected)
+					gtk_html_draw_bit(html, hb, 1);
+				hbits = hbits->prev;
+			}
+			hbits = g_list_last(html->html_bits);
+		}
+        }
+
+        return TRUE;
+}
+
+
+
+static void gtk_html_selection_get (GtkWidget *widget,
+                                    GtkSelectionData  *selection_data,
+                                    guint sel_info,
+                                    guint32 time)
+{
+	gchar *str;
+        gint len;
+        GtkHtml *html;
+
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+
+        html = GTK_HTML(widget);
+
+        
+	if (selection_data->selection != GDK_SELECTION_PRIMARY)
+		return;
+
+	str = html->selected_text;
+
+	if (!str)
+		return;
+	
+	len = strlen(str);
+
+	if (sel_info == TARGET_STRING) {
+		gtk_selection_data_set(selection_data,
+				       GDK_SELECTION_TYPE_STRING,
+				       8*sizeof(gchar), (guchar *)str, len);
+	} else if ((sel_info == TARGET_TEXT) || (sel_info == TARGET_COMPOUND_TEXT)) {
+		guchar *text;
+		GdkAtom encoding;
+		gint format;
+		gint new_length;
+
+		gdk_string_to_compound_text (str, &encoding, &format, &text, &new_length);
+		gtk_selection_data_set (selection_data, encoding, format, text, new_length);
+		gdk_free_compound_text (text);
+	}
+
+
+
+}
+
+static void do_select(GtkHtml *html,
+                      int x,
+                      int y)
+{
+	GList *hbits = g_list_last(html->html_bits);
+	int epos, spos;
+	GtkHtmlBit *hb;
+	
+	if (!hbits)
+		return;
+	
+        hb = (GtkHtmlBit *)hbits->data;
+
+        while (hbits) {
+                hb = (GtkHtmlBit *)hbits->data;
+                if (hb->type == HTML_BIT_TEXT)
+			break;
+		hbits = hbits->prev;
+        }
+        
+	if (!hb)
+                return;
+
+
+	if (y > hb->y) {
+		html->num_end = strlen(hb->text) - 1;
+		html->end_sel = hb;
+	} else if (y < 0) {
+		html->num_end = 0;
+		html->end_sel = (GtkHtmlBit *)html->html_bits->data;
+	} else 	while(hbits) {
+		hb = (GtkHtmlBit *)hbits->data;
+		if ((y < hb->y && y > (hb->y - hb->height)) &&
+		    (x > hb->x + hb->width)) {
+			if (hb->type != HTML_BIT_TEXT) {
+				html->num_end = 0;
+				html->end_sel = hb;
+				break;
+			}
+
+			html->num_end = strlen(hb->text) - 1;
+			html->end_sel = hb;
+			break;
+		} else if ((x > hb->x && x < (hb->x + hb->width)) &&
+			   (y < hb->y && y > (hb->y - hb->height))) {
+			int i, len;
+			int w = x - hb->x;
+
+			if (hb->type != HTML_BIT_TEXT) {
+				html->num_end = 0;
+				html->end_sel = hb;
+				break;
+			}
+
+                        len = strlen(hb->text);
+                        
+			for (i=1; i<=len; i++) {
+				if (gdk_text_measure(hb->font, hb->text, i) > w) {
+					html->num_end = i - 1;
+					html->end_sel = hb;
+					break;
+				}
+			}
+			break;
+		}
+		hbits = hbits->prev;
+	}
+
+	if (html->end_sel == NULL)
+		return;
+	if (html->start_sel == NULL) {
+		html->start_sel = html->end_sel;
+		html->num_start = html->num_end;
+	}
+	
+	epos = g_list_index(html->html_bits, html->end_sel);
+	spos = g_list_index(html->html_bits, html->start_sel);
+	g_free(html->selected_text);
+	html->selected_text = NULL;
+
+	if (epos == spos) {
+		char *str;
+		if (html->start_sel->type != HTML_BIT_TEXT) {
+			html->selected_text = NULL;
+			return;
+		}
+		if (html->num_end == html->num_start) {
+			str = g_malloc(2);
+                        if (strlen(html->start_sel->text))
+				str[0] = html->start_sel->text[html->num_end];
+			else
+                                str[0] = 0;
+			str[1] = 0;
+			gtk_html_draw_bit(html, html->start_sel, 0);
+			html->selected_text = str;
+		} else {
+			int st, en;
+			char *str;
+			if (html->num_end > html->num_start) {
+				en = html->num_end;
+				st = html->num_start;
+			} else {
+				en = html->num_start;
+				st = html->num_end;
+			}
+
+			str = g_malloc(en - st + 2);
+			strncpy(str, html->start_sel->text + st, (en - st + 1));
+			str[en - st + 1] = 0;
+                        gtk_html_draw_bit(html, html->start_sel, 0);
+			html->selected_text = str;
+			
+		}
+	} else {
+		GtkHtmlBit *shb, *ehb;
+		int en, st;
+		int len, nlen;
+		char *str;
+		if (epos > spos) {
+			shb = html->start_sel;
+			ehb = html->end_sel;
+			en = html->num_end;
+			st = html->num_start;
+		} else {
+			shb = html->end_sel;
+			ehb = html->start_sel;
+			en = html->num_start;
+			st = html->num_end;
+		}
+		
+		hbits = g_list_find(html->html_bits, shb);
+
+		if (!hbits)
+			return;
+		
+		if (shb->type == HTML_BIT_TEXT) {
+			len = strlen(shb->text) - st + 1;
+			str = g_malloc(len);
+			strcpy(str, shb->text + st);
+			str[len - 1] = 0;
+			gtk_html_draw_bit(html, shb, 0);
+			if (shb->newline) {
+				len+= 1;
+				str = g_realloc(str, len);
+				str[len - 2] = '\n';
+				str[len - 1] = 0;
+			}
+		} else {
+			len = 1;
+			str = g_malloc(1);
+			str[0] = 0;
+		}
+		if (hbits->next == NULL) {
+			html->selected_text = str;
+			return;
+		}
+
+		
+                hbits = hbits->next;
+		while(1) { /* Yah I know is dangerous :P */
+			hb = (GtkHtmlBit *)hbits->data;
+			if (hb->type != HTML_BIT_TEXT) {
+				if (hb == ehb)
+					break;
+				hbits = hbits->next;
+				continue;
+			}
+			if (hb != ehb) {
+				nlen = len + strlen(hb->text);
+				str = g_realloc(str, nlen);
+				strcpy(str + (len - 1), hb->text);
+				len = nlen;
+				str[len - 1] = 0;
+				gtk_html_draw_bit(html, hb, 0);
+				if (hb->newline) {
+					len+= 1;
+					str = g_realloc(str, len);
+					str[len - 2] = '\n';
+					str[len - 1] = 0;
+				}
+			} else {
+				nlen = len + en + 1;
+				str = g_realloc(str, nlen);
+				strncpy(str + (len - 1), hb->text, en + 1);
+				len = nlen;
+				str[len - 1] = 0;
+				
+				gtk_html_draw_bit(html, hb, 0);
+				if (hb->newline && en == strlen(hb->text)) {
+					len+= 1;
+					str = g_realloc(str, len);
+					str[len - 2] = '\n';
+					str[len - 1] = 0;
+				}
+				break;
+			}
+			hbits = hbits->next;
+		}
+		html->selected_text = str;
+	}
+
+}
+
+static gint
+scroll_timeout(GtkHtml *html)
+{
+        GdkEventMotion event;
+        gint x, y;
+        GdkModifierType mask;
+
+	html->timer = 0;
+	gdk_window_get_pointer (html->html_area, &x, &y, &mask);
+  
+	if (mask & GDK_BUTTON1_MASK)
+	{
+		event.is_hint = 0;
+		event.x = x;
+		event.y = y;
+		event.state = mask;
+
+                gtk_html_motion_notify (GTK_WIDGET (html), &event);
+	}
+
+	return FALSE;
+
+}
+
+
+static gint gtk_html_tooltip_paint_window(GtkHtml *html)
+{
+	GtkStyle *style;
+	gint y, baseline_skip, gap;
+
+	style = html->tooltip_window->style;
+
+	gap = (style->font->ascent + style->font->descent) / 4;
+	if (gap < 2)
+		gap = 2;
+	baseline_skip = style->font->ascent + style->font->descent + gap;
+
+	if (!html->tooltip_hb)
+		return FALSE;
+
+	gtk_paint_flat_box(style, html->tooltip_window->window,
+			   GTK_STATE_NORMAL, GTK_SHADOW_OUT,
+			   NULL, GTK_WIDGET(html->tooltip_window), "tooltip",
+			   0, 0, -1, -1);
+
+	y = style->font->ascent + 4;
+
+	gtk_paint_string (style, html->tooltip_window->window,
+			  GTK_STATE_NORMAL,
+			  NULL, GTK_WIDGET(html->tooltip_window), "tooltip",
+			  4, y, "HTML Link:");
+	y += baseline_skip;
+	gtk_paint_string (style, html->tooltip_window->window,
+			  GTK_STATE_NORMAL,
+			  NULL, GTK_WIDGET(html->tooltip_window), "tooltip",
+			  4, y, html->tooltip_hb->url);
+	
+	return FALSE;
+
+
+}
+
+static gint gtk_html_tooltip_timeout(gpointer data)
+{
+	GtkHtml *html = (GtkHtml *)data;
+
+	
+	GDK_THREADS_ENTER();
+
+	if (html->tooltip_hb && GTK_WIDGET_DRAWABLE(GTK_WIDGET(html))) {
+		GtkWidget *widget;
+		GtkStyle *style;
+		gint gap, x, y, w, h, scr_w, scr_h, baseline_skip;
+
+		if (html->tooltip_window)
+			gtk_widget_destroy(html->tooltip_window);
+		
+		html->tooltip_window = gtk_window_new (GTK_WINDOW_POPUP);
+		gtk_widget_set_app_paintable (html->tooltip_window, TRUE);
+		gtk_window_set_policy (GTK_WINDOW (html->tooltip_window), FALSE, FALSE, TRUE);
+		gtk_widget_set_name (html->tooltip_window, "gtk-tooltips");
+		gtk_signal_connect_object (GTK_OBJECT (html->tooltip_window),
+					   "expose_event",
+					   GTK_SIGNAL_FUNC (gtk_html_tooltip_paint_window),
+					   GTK_OBJECT (html));
+		gtk_signal_connect_object (GTK_OBJECT (html->tooltip_window),
+					   "draw",
+					   GTK_SIGNAL_FUNC (gtk_html_tooltip_paint_window),
+					   GTK_OBJECT (html));
+		
+		gtk_widget_ensure_style (html->tooltip_window);
+		style = html->tooltip_window->style;
+  
+		widget = GTK_WIDGET(html);
+
+		scr_w = gdk_screen_width ();
+		scr_h = gdk_screen_height ();
+
+		gap = (style->font->ascent + style->font->descent) / 4;
+		if (gap < 2)
+			gap = 2;
+		baseline_skip = style->font->ascent + style->font->descent + gap;
+
+		w = 8 + MAX(gdk_string_width(style->font, "HTML Link:"),
+			    gdk_string_width(style->font, html->tooltip_hb->url));
+					     ;
+		h = 8 - gap;
+      		h += (baseline_skip * 2);
+
+		gdk_window_get_pointer (NULL, &x, &y, NULL);
+		/*gdk_window_get_origin (widget->window, NULL, &y);*/
+		if (GTK_WIDGET_NO_WINDOW (widget))
+			y += widget->allocation.y;
+
+		x -= ((w >> 1) + 4);
+
+		if ((x + w) > scr_w)
+			x -= (x + w) - scr_w;
+		else if (x < 0)
+			x = 0;
+
+		if ((y + h + 4) > scr_h)
+			y = y - html->tooltip_hb->font->ascent + html->tooltip_hb->font->descent;
+		else
+			y = y + html->tooltip_hb->font->ascent + html->tooltip_hb->font->descent;
+
+		gtk_widget_set_usize (html->tooltip_window, w, h);
+		gtk_widget_popup (html->tooltip_window, x, y);
+		
+	}
+
+	html->tooltip_timer = -1;
+	
+	GDK_THREADS_LEAVE();
+
+	return FALSE;
+}
+
+
+static gint gtk_html_leave_notify (GtkWidget      *widget,
+                                    GdkEventCrossing *event)
+{
+       GtkHtml *html;
+
+       html = GTK_HTML(widget);
+
+       if (html->tooltip_timer != -1)
+	       gtk_timeout_remove(html->tooltip_timer);
+       if (html->tooltip_window) {
+	       gtk_widget_destroy(html->tooltip_window);
+	       html->tooltip_window = NULL;
+       }
+
+
+       html->tooltip_hb = NULL;
+       return TRUE;
+}
+
+
+static gint gtk_html_motion_notify (GtkWidget *widget,
+                                    GdkEventMotion *event)
+{
+        int x, y;
+        gint width, height;
+        GdkModifierType state;
+        int realx, realy;
+        GtkHtml *html = GTK_HTML(widget);
+
+        if (event->is_hint)
+                gdk_window_get_pointer (event->window, &x, &y, &state);
+        else
+        {
+                x = event->x;
+                y = event->y;
+                state = event->state;
+        }
+
+        gdk_window_get_size(html->html_area, &width, &height);
+        
+	realx = x;
+	realy = y + html->yoffset;
+
+
+	if (state & GDK_BUTTON1_MASK) {
+		if (realx != html->start_sel_x || realy != html->start_sel_y) {
+			char *tmp = NULL;
+
+			if (y < 0 || y > height) {
+				int diff;
+				if (html->timer == 0) {
+					html->timer = gtk_timeout_add(100,
+								    (GtkFunction)scroll_timeout,
+								    html);
+					if (y < 0)
+						diff = y / 2;
+					else
+						diff = (y - height) / 2;
+
+					if (html->vadj->value + diff >
+					    html->vadj->upper - height + 20)
+						gtk_adjustment_set_value(html->vadj,
+									 html->vadj->upper - height + 20);
+					else
+						gtk_adjustment_set_value(html->vadj,
+									 html->vadj->value + diff);
+
+				}
+			}
+			
+			if (html->selected_text != NULL)
+				tmp = g_strdup(html->selected_text);
+			do_select(html, realx, realy);
+			if (tmp) {
+				if (!html->selected_text || strcmp(tmp, html->selected_text)) {
+					GtkHtmlBit *hb;
+					GList *hbits = html->html_bits;
+					while(hbits) {
+						hb = (GtkHtmlBit *)hbits->data;
+						if (hb->was_selected)
+							gtk_html_draw_bit(html, hb, 0);
+						hbits = hbits->next;
+					}
+				}
+				g_free(tmp);
+			}
+		}
+	} else {
+		GtkHtmlBit *hb;
+		GList *urls;
+
+		urls = html->urls;
+		while(urls) {
+                        hb = (GtkHtmlBit *)urls->data;
+			if ((realx > hb->x && realx < (hb->x + hb->width)) &&
+			    (realy < hb->y && realy > (hb->y - hb->height))) {
+				if (html->tooltip_hb != hb) {
+					html->tooltip_hb = hb;
+					if (html->tooltip_timer != -1)
+						gtk_timeout_remove(html->tooltip_timer);
+					if (html->tooltip_window) {
+						gtk_widget_destroy(html->tooltip_window);
+						html->tooltip_window = NULL;
+					}
+					html->tooltip_timer = gtk_timeout_add(HTML_TOOLTIP_DELAY, gtk_html_tooltip_timeout, html);
+				}
+				gdk_window_set_cursor(html->html_area, gdk_cursor_new(GDK_HAND2));
+				return TRUE;
+			}
+                        urls = urls->next;
+		}
+		if (html->tooltip_timer != -1)
+			gtk_timeout_remove(html->tooltip_timer);
+		if (html->tooltip_window) {
+			gtk_widget_destroy(html->tooltip_window);
+			html->tooltip_window = NULL;
+		}
+			
+			
+                html->tooltip_hb = NULL;
+		gdk_window_set_cursor(html->html_area, NULL);
+
+
+	}
+
+	return TRUE;
+}
+
+static gint gtk_html_button_release (GtkWidget *widget,
+                                     GdkEventButton *event)
+{
+        GtkHtml *html;
+
+        html = GTK_HTML(widget);
+
+        if (html->frozen > 0)
+                return TRUE;
+        
+        if (event->button == 1) {
+                int realx, realy;
+                GtkHtmlBit *hb;
+                GList *urls = html->urls;
+
+                realx = event->x;
+                realy = event->y + html->yoffset;
+                if (realx != html->start_sel_x || realy != html->start_sel_y) {
+			if (gtk_selection_owner_set (widget,
+						 GDK_SELECTION_PRIMARY,
+                                                     event->time)) {
+                        } else {
+                        }
+		} else {
+			if (gdk_selection_owner_get(GDK_SELECTION_PRIMARY) == widget->window)
+				gtk_selection_owner_set(NULL, GDK_SELECTION_PRIMARY,
+					 event->time);
+
+		
+			while(urls) {
+				hb = (GtkHtmlBit *)urls->data;
+				if ((realx > hb->x && realx < (hb->x + hb->width)) &&
+                                    (realy < hb->y && realy > (hb->y - hb->height))) {
+                                        if (web_browser == BROWSER_NETSCAPE &&
+                                           (general_options & OPT_GEN_BROWSER_POPUP))
+                                                open_url_nw(NULL, hb->url);
+                                        else
+                                                open_url(NULL, hb->url);
+					break;
+				}
+				urls = urls->next;
+			}
+		}
+	}
+	return TRUE;
+}
+
+
+
+static gint gtk_html_button_press (GtkWidget *widget,
+                            GdkEventButton *event)
+{
+        GtkHtml *html;
+        gfloat value;
+
+
+        html = GTK_HTML(widget);
+        value = html->vadj->value;
+
+        if (html->frozen > 0)
+                return TRUE;
+	
+	if (event->button == 4) {
+		value -= html->vadj->step_increment;
+		if (value < html->vadj->lower)
+			value = html->vadj->lower;
+		gtk_adjustment_set_value(html->vadj,
+					 value);
+	} else if (event->button == 5) {
+		value += html->vadj->step_increment;
+		if (value > html->vadj->upper)
+			value = html->vadj->upper;
+		gtk_adjustment_set_value(html->vadj,
+					 value);
+
+        } else if (event->button == 1) {
+                GList *hbits = g_list_last(html->html_bits);
+		int realx, realy;
+		GtkHtmlBit *hb;
+
+		realx = event->x;
+		realy = event->y + html->yoffset;
+
+		html->start_sel_x = realx;
+		html->start_sel_y = realy;
+
+		if (!hbits)
+			return TRUE;
+
+		if (html->selected_text) {
+			g_free(html->selected_text);
+			html->selected_text = NULL;
+			html->start_sel = NULL;
+			html->end_sel = NULL;
+			html->num_start = 0;
+			html->num_end = 0;
+			while(hbits) {
+				hb = (GtkHtmlBit *)hbits->data;
+				if (hb->was_selected)
+					gtk_html_draw_bit(html, hb, 1);
+				hbits = hbits->prev;
+			}
+			hbits = g_list_last(html->html_bits);
+		}
+
+                hb = (GtkHtmlBit *)hbits->data;
+		if (realy > hb->y) {
+			if (hb->text)
+				html->num_start = strlen(hb->text) - 1;
+			else
+                                html->num_start = 0;
+			html->start_sel = hb;
+		} else 	while(hbits) {
+			hb = (GtkHtmlBit *)hbits->data;
+			if ((realy < hb->y && realy > (hb->y - hb->height)) &&
+			    (realx > hb->x + hb->width)) {
+				if (hb->type != HTML_BIT_TEXT) {
+					html->num_end = 0;
+					html->end_sel = hb;
+					break;
+				}
+
+                                if (hb->text)
+					html->num_start = strlen(hb->text) - 1;
+				else
+                                        html->num_start = 0;
+                                
+				html->start_sel = hb;
+				break;
+			} else if ((realx > hb->x && realx < (hb->x + hb->width)) &&
+				   (realy < hb->y && realy > (hb->y - hb->height))) {
+				int i, len;
+				int w = realx - hb->x;
+
+				if (hb->type != HTML_BIT_TEXT) {
+					html->num_end = 0;
+					html->end_sel = hb;
+					break;
+				}
+
+				if (hb->text)
+					len = strlen(hb->text);
+				else
+					len = 0;
+				
+				for (i=1; i<=len; i++) {
+					if (gdk_text_measure(hb->font, hb->text, i) > w) {
+						html->num_start = i - 1;
+						html->start_sel = hb;
+						break;
+					}
+				}
+				break;
+			}
+			hbits = hbits->prev;
+                }
+	} else if (event->button == 3 && event->type == GDK_BUTTON_PRESS) {
+		GtkHtmlBit *hb = NULL;
+		int realx, realy;
+	        GList *urls;
+	                                
+		realx = event->x;
+		realy = event->y + html->yoffset;
+		                                                                
+		urls = html->urls;
+		while(urls) {
+                        hb = (GtkHtmlBit *)urls->data;
+			if ((realx > hb->x && realx < (hb->x + hb->width)) &&
+			    (realy < hb->y && realy > (hb->y - hb->height))) {
+				break;
+			}
+                        urls = urls->next;
+			hb = NULL;
+		}
+		
+		if (hb != NULL) {
+		        GtkWidget *menu, *button;
+	
+        	        menu = gtk_menu_new();
+
+                        if (web_browser == BROWSER_NETSCAPE) {
+                        
+                                button = gtk_menu_item_new_with_label("Open URL in existing window");
+                                gtk_signal_connect(GTK_OBJECT(button), "activate",
+                                                   GTK_SIGNAL_FUNC(open_url), hb->url);
+                                gtk_menu_append(GTK_MENU(menu), button);
+                                gtk_widget_show(button);
+
+                        }
+
+	
+                        button = gtk_menu_item_new_with_label("Open URL in new window");
+                        gtk_signal_connect(GTK_OBJECT(button), "activate",
+                                           GTK_SIGNAL_FUNC(open_url_nw), hb->url);
+                        gtk_menu_append(GTK_MENU(menu), button);
+                        gtk_widget_show(button);
+
+                        if (web_browser == BROWSER_NETSCAPE) {
+
+                                button = gtk_menu_item_new_with_label("Add URL as bookmark");
+                                gtk_signal_connect(GTK_OBJECT(button), "activate",
+                                                   GTK_SIGNAL_FUNC(add_bookmark), hb->url);
+                                gtk_menu_append(GTK_MENU(menu), button);
+                                gtk_widget_show(button);
+
+                        }
+                	
+	                gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL,
+                                       event->button, event->time);                	
+		}		
+	}
+	
+	return TRUE;
+}
+
+
+static void gtk_html_draw_bit(GtkHtml *html,
+                              GtkHtmlBit *hb,
+                              int redraw)
+{
+        int mypos, epos, spos;
+        GdkGC *gc = html->gc;
+	int shift;
+        GtkStateType selected_state;
+        GtkWidget *widget = GTK_WIDGET(html);
+	GdkRectangle area;
+
+	if (html->frozen > 0)
+		return;
+
+	if (hb->type == HTML_BIT_TEXT) {
+
+		if (!strlen(hb->text))
+			return;
+		
+		mypos = g_list_index(html->html_bits, hb);
+		epos = g_list_index(html->html_bits, html->end_sel);
+		spos = g_list_index(html->html_bits, html->start_sel);
+
+                if (((html->end_sel == NULL) || (html->start_sel == NULL)) ||
+                    ((epos < mypos) && (spos < mypos)) ||
+                    ((epos > mypos) && (spos > mypos))) {
+                        selected_state = GTK_STATE_NORMAL;
+		} else {
+			selected_state = GTK_STATE_SELECTED;
+		}
+
+
+		gdk_text_extents(hb->font, hb->text, 1, &shift, NULL, NULL, NULL, NULL);
+
+                if (selected_state == GTK_STATE_SELECTED) {
+                        int schar = 0, echar = 0;
+			int startx = 0, xwidth = 0;
+
+                        if (epos > spos ||
+                            (epos == spos && html->num_end >= html->num_start)) {
+				if (mypos == epos) {
+					echar = html->num_end;
+					xwidth = gdk_text_measure(hb->font, hb->text, html->num_end + 1);
+				} else {
+					echar = strlen(hb->text);
+					xwidth = hb->width;
+				}
+				if (mypos == spos) {
+					schar = html->num_start;
+					startx = gdk_text_measure(hb->font, hb->text, html->num_start);
+					xwidth -= startx;
+				}
+			} else {
+				if (mypos == spos) {
+					echar = html->num_start;
+					xwidth = gdk_text_measure(hb->font, hb->text, html->num_start + 1);
+				} else {
+					echar = strlen(hb->text);
+					xwidth = hb->width;
+				}
+				if (mypos == epos) {
+					schar = html->num_end;
+					startx = gdk_text_measure(hb->font, hb->text, html->num_end);
+					xwidth -= startx;
+				}
+			}
+
+			if (!redraw && echar == hb->sel_e && schar == hb->sel_s)
+				return;
+				
+			hb->sel_e = echar;
+			hb->sel_s = schar;
+			
+			startx += hb->x;
+
+
+                        area.x = hb->x - html->xoffset;
+                        area.y = hb->y - hb->height + 3 - html->yoffset;
+                        area.width = hb->width+2;
+                        area.height = hb->height;
+                        clear_area (html, &area);
+
+			gtk_paint_flat_box (widget->style, html->html_area,
+					    selected_state, GTK_SHADOW_NONE,
+					    NULL, widget, "text",
+					    startx,
+					    hb->y - hb->height + 3 - html->yoffset,
+					    xwidth+2, hb->height);
+                        hb->was_selected = 1;
+                } else if (hb->was_selected) {
+                        area.x = hb->x - html->xoffset;
+                        area.y = hb->y - hb->height + 3 - html->yoffset;
+                        area.width = hb->width+2;
+                        area.height = hb->height;
+                        clear_area (html, &area);
+
+                        hb->sel_e = -1;
+                        hb->sel_s = -1;
+			
+			hb->was_selected = 0;
+		}
+		
+		
+
+
+                if (selected_state == GTK_STATE_SELECTED && (mypos == epos
+		   || mypos == spos)) {
+			char *s = hb->text;
+			int num = 0, width = 0, fsel = 0, esel = strlen(hb->text);
+			int lbearing, rbearing, w;
+			
+                        if (epos > spos ||
+                            (epos == spos && html->num_end >= html->num_start)) {
+				if (mypos == epos)
+					esel = html->num_end;
+				if (mypos == spos)
+					fsel = html->num_start;
+			} else {
+				if (mypos == spos)
+					esel = html->num_start;
+                                if (mypos == epos)
+                                        fsel = html->num_end;
+			}
+
+			while(*s) {
+
+				if (num < fsel || num > esel)
+					selected_state = GTK_STATE_NORMAL;
+				else
+					selected_state = GTK_STATE_SELECTED;
+				if (hb->fore != NULL)
+					gdk_gc_set_foreground(gc, hb->fore);
+				else
+					gdk_gc_set_foreground(gc, &widget->style->text[selected_state]);
+				if (hb->back != NULL)
+					gdk_gc_set_background(gc, hb->back);
+				else
+					gdk_gc_set_background(gc, &widget->style->bg[selected_state]);
+
+
+				gdk_gc_set_font(gc, hb->font);
+
+				gdk_text_extents(hb->font, s, 1, &lbearing, &rbearing, &w, NULL, NULL);
+
+                                gdk_draw_text(html->html_area, hb->font, gc, shift + hb->x  + width, hb->y - html->yoffset, s, 1);
+
+                                if (hb->uline)
+					gdk_draw_line(html->html_area, gc, shift + hb->x  + width, hb->y - html->yoffset, shift + hb->x + width + w, hb->y - html->yoffset);
+
+				if (hb->strike)
+					gdk_draw_line(html->html_area, gc, shift + hb->x  + width, hb->y - html->yoffset - (hb->height / 3), shift + hb->x + width + w, hb->y - html->yoffset - (hb->height / 3));
+
+				width += w;
+				
+				s++;
+				num++;
+			}
+
+
+		} else {
+
+			if (hb->fore != NULL)
+				gdk_gc_set_foreground(gc, hb->fore);
+			else
+				gdk_gc_set_foreground(gc, &widget->style->text[selected_state]);
+			if (hb->back != NULL)
+				gdk_gc_set_background(gc, hb->back);
+			else
+				gdk_gc_set_background(gc, &widget->style->bg[selected_state]);
+
+
+			gdk_gc_set_font(gc, hb->font);
+
+			gdk_draw_string(html->html_area, hb->font, gc, shift + hb->x , hb->y - html->yoffset, hb->text);
+			if (hb->uline)
+				gdk_draw_line(html->html_area, gc, shift + hb->x , hb->y - html->yoffset, hb->x + gdk_string_measure(hb->font, hb->text), hb->y - html->yoffset);
+
+			if (hb->strike)
+				gdk_draw_line(html->html_area, gc, shift + hb->x , hb->y - html->yoffset - (hb->height / 3), hb->x + gdk_string_measure(hb->font, hb->text), hb->y - html->yoffset - (hb->height / 3));
+
+		}
+	} else if (hb->type == HTML_BIT_SEP) {
+
+		gdk_draw_line(html->html_area, gc, hb->x + 2 , hb->y - html->yoffset - (hb->height / 2 - 1), hb->x + hb->width, hb->y - html->yoffset - (hb->height / 2 - 1));
+
+	} else if (hb->type == HTML_BIT_PIXMAP) {
+		gdk_gc_set_background(gc, &widget->style->base[GTK_STATE_NORMAL]);
+		gdk_draw_pixmap(html->html_area, gc, hb->pm, 0, 0, hb->x , hb->y - html->yoffset - (hb->height) + 4, hb->width, hb->height - 2);
+	}
+}
+
+
+
+gint compare_types(GtkHtmlBit *hb, GtkHtmlBit *hb2)
+{
+	/* In this function, it's OK to accidently return a
+	 * 0, but will cause problems on an accidental 1 */
+
+	if (!hb || !hb2)
+		return 0;
+	
+	
+	if (hb->uline != hb2->uline)
+		return 0;
+	if (hb->strike != hb2->strike)
+                return 0;
+	if (hb->font && hb2->font) {
+		if (!gdk_font_equal(hb->font, hb2->font))
+			return 0;
+	} else if (hb->font && !hb2->font) {
+		return 0;
+	} else if (!hb->font && hb2->font) {
+		return 0;
+	}
+	if (hb->type != hb2->type)
+		return 0;
+	
+	if (hb->fore && hb2->fore) {
+		if (!gdk_color_equal(hb->fore, hb2->fore))
+			return 0;
+	} else if (hb->fore && !hb2->fore) {
+		return 0;
+	} else if (!hb->fore && hb2->fore) {
+		return 0;
+	}
+
+	if (hb->back && hb2->back) {
+		if (!gdk_color_equal(hb->back, hb2->back))
+			return 0;
+	} else if (hb->back && !hb2->back) {
+		return 0;
+	} else if (!hb->back && hb2->back) {
+		return 0;
+	}
+
+	if ((hb->url != NULL && hb2->url == NULL) ||
+	    (hb->url == NULL && hb2->url != NULL))
+		return 0;
+	
+        if (hb->url != NULL && hb2->url != NULL)
+		if (strcasecmp(hb->url, hb2->url))
+			return 0;
+	
+	return 1;
+}
+
+static gint html_bit_is_onscreen(GtkHtml *html, GtkHtmlBit *hb)
+{
+	gint width, height;
+
+	gdk_window_get_size(html->html_area, &width, &height);
+	
+	if (hb->y < html->yoffset) {
+		return 0;
+	}
+
+	if ((hb->y - hb->height) > (html->yoffset + height)) {
+		return 0;
+	}
+	return 1;
+}
+
+static void draw_cursor(GtkHtml *html)
+{
+	if (
+	    html->editable &&
+	    html->cursor_hb &&
+	    GTK_WIDGET_DRAWABLE(html) &&
+	    html_bit_is_onscreen(html, html->cursor_hb)) {
+		gint x, y;
+		gint width;
+
+		GdkFont *font = html->cursor_hb->font;
+
+		gdk_text_extents(font, html->cursor_hb->text, html->cursor_pos, NULL, NULL, &width, NULL, NULL);
+
+		gdk_gc_set_foreground(html->gc, &GTK_WIDGET(html)->style->text[GTK_STATE_NORMAL]);
+
+		y = html->cursor_hb->y - html->yoffset;
+		x = html->cursor_hb->x + width;
+
+
+		gdk_draw_line (html->html_area, html->gc, x,
+			       y, x, y - font->ascent);
+
+	}
+}
+
+static void undraw_cursor(GtkHtml *html)
+{
+	if (
+	    html->editable &&
+	    html->cursor_hb &&
+	    GTK_WIDGET_DRAWABLE(html) &&
+	    html_bit_is_onscreen(html, html->cursor_hb)) {
+		gint x, y;
+		gint width;
+                GdkRectangle area;
+		
+		GdkFont *font = html->cursor_hb->font;
+
+		gdk_text_extents(font, html->cursor_hb->text, html->cursor_pos, NULL, NULL, &width, NULL, NULL);
+
+		y = html->cursor_hb->y - html->yoffset;
+		x = html->cursor_hb->x + width;
+
+		area.x = x;
+		area.y = y - font->ascent;
+		area.height = font->ascent + 1;
+		area.width = 1;
+
+
+		clear_area (html, &area);
+
+		gtk_html_draw_bit(html, html->cursor_hb, 1);
+		
+
+	    }
+}
+
+
+static void expose_html(GtkHtml *html,
+                        GdkRectangle *area,
+                        gboolean cursor)
+{
+        GList *hbits;
+        GtkHtmlBit *hb;
+        gint width, height;
+        gint realy;
+
+        
+        if (html->frozen > 0)
+        	return;
+
+        
+        hbits = html->html_bits;
+
+	gdk_window_get_size(html->html_area, &width, &height);
+
+        realy = area->y + html->yoffset;
+
+        clear_area (html, area);
+
+        while(hbits) {
+                
+                hb = (GtkHtmlBit *)hbits->data;
+
+		if (html_bit_is_onscreen(html, hb))
+                        gtk_html_draw_bit(html, hb, 1);
+
+
+                hbits = hbits->next;
+        }
+}
+
+static void resize_html(GtkHtml *html)
+{
+	GList *hbits = html->html_bits;
+	GList *html_bits = html->html_bits;
+	GtkHtmlBit *hb, *hb2;
+	char *str;
+	gint height;
+
+        if(!hbits)
+                return;
+        
+        
+        html->html_bits = NULL;
+
+        html->current_x = 0;
+	html->current_y = 0;
+
+        html->vadj->upper = 0;
+	
+        gtk_html_freeze(html);
+	
+	while(hbits) {
+		hb = (GtkHtmlBit *)hbits->data;
+		if (hb->type == HTML_BIT_SEP) {
+			
+			gtk_html_add_seperator(html);
+
+			g_free(hb);
+
+			hbits = hbits->next;
+			continue;
+		}
+		if (hb->type == HTML_BIT_PIXMAP) {
+
+			gtk_html_add_pixmap(html, hb->pm, hb->fit);
+
+			g_free(hb);
+
+			hbits = hbits->next;
+			continue;
+		}
+		
+		if (hb->newline) {
+			int i;
+
+			if (!hb->text) {
+				hb->text = g_malloc(1);
+				hb->text[0] = 0;
+			}
+			for (i=0; i<hb->newline; i++) {
+                                str = hb->text;
+				hb->text = g_strconcat(str, "\n", NULL);
+				g_free(str);
+			}
+		}
+
+		if (hbits->next) {
+			hb2 = (GtkHtmlBit *)hbits->next->data;
+                } else {
+                        hb2 = NULL;
+                }
+
+
+		
+		if (!hb->newline && compare_types(hb, hb2)) {
+			str = hb2->text;
+			hb2->text = g_strconcat(hb->text, hb2->text, NULL);
+			g_free(str);
+			hb2 = NULL;
+		} else if (hb->text) {
+			gtk_html_add_text(html, hb->font, hb->fore, hb->back,
+				 hb->text, strlen(hb->text), hb->uline, hb->strike, hb->url);
+		}
+
+		
+
+		/* Font stays, so do colors (segfaults if I free) */
+		if (hb->fore)
+			gdk_color_free(hb->fore);
+		if (hb->back)
+			gdk_color_free(hb->back);
+		if (hb->text)
+			g_free(hb->text);
+		if (hb->url)
+			g_free(hb->url);
+
+                g_free(hb);
+
+		hbits = hbits->next;
+	}
+
+        g_list_free(html_bits);
+
+
+        gtk_html_thaw(html);
+        
+	gdk_window_get_size(html->html_area, NULL, &height);
+        gtk_adjustment_set_value(html->vadj, html->vadj->upper - height);
+
+}
+
+static GdkGC *create_bg_gc (GtkHtml *html)
+{
+        GdkGCValues values;
+
+        values.tile = GTK_WIDGET (html)->style->bg_pixmap[GTK_STATE_NORMAL];
+        values.fill = GDK_TILED;
+
+        return gdk_gc_new_with_values (html->html_area, &values,
+                                       GDK_GC_FILL | GDK_GC_TILE);
+}
+
+static void clear_area (GtkHtml *html,
+                        GdkRectangle *area)
+{
+	GtkWidget *widget = GTK_WIDGET (html);
+	gint x, y;
+	
+	
+	if (html->transparent) {
+		if (html->pm == NULL)
+			html->pm = get_desktop_pixmap(widget);
+
+                if (html->pm == NULL)
+                        return;
+
+                if (html->bg_gc == NULL) {
+                        GdkGCValues values;
+
+                        values.tile = html->pm;
+                        values.fill = GDK_TILED;
+
+                        html->bg_gc = gdk_gc_new_with_values (html->html_area, &values,
+                                                       GDK_GC_FILL | GDK_GC_TILE);
+
+                }
+                        
+		gdk_window_get_deskrelative_origin(html->html_area, &x, &y);
+
+                gdk_draw_pixmap(html->html_area, html->bg_gc, html->pm,
+                                x + area->x, y + area->y, area->x, area->y, area->width,
+                                area->height);
+
+		return;
+
+	}
+        if (html->bg_gc) {
+
+                gint width, height;
+      
+                gdk_window_get_size (widget->style->bg_pixmap[GTK_STATE_NORMAL], &width, &height);
+      
+                gdk_gc_set_ts_origin (html->bg_gc,
+                                      (- html->xoffset) % width,
+                                      (- html->yoffset) % height);
+
+                gdk_draw_rectangle (html->html_area, html->bg_gc, TRUE,
+                                    area->x, area->y, area->width, area->height);
+        }
+        else
+                gdk_window_clear_area (html->html_area, area->x, area->y, area->width, area->height);
+}
+
+
+
+
+static void gtk_html_destroy (GtkObject *object)
+{
+        GtkHtml *html;
+
+        g_return_if_fail(object != NULL);
+        g_return_if_fail(GTK_IS_HTML (object));
+
+        html = (GtkHtml *)object;
+
+
+        gtk_signal_disconnect_by_data (GTK_OBJECT (html->hadj), html);
+        gtk_signal_disconnect_by_data (GTK_OBJECT (html->vadj), html);
+
+        if (html->timer) {
+                gtk_timeout_remove (html->timer);
+                html->timer = 0;
+	}
+
+	if (html->tooltip_timer) {
+		gtk_timeout_remove (html->tooltip_timer);
+		html->tooltip_timer = -1;
+	}
+
+        
+        GTK_OBJECT_CLASS(parent_class)->destroy(object);
+        
+}
+
+static void gtk_html_finalize (GtkObject *object)
+{
+        GList *hbits;
+        GtkHtml *html;
+	GtkHtmlBit *hb;
+
+
+        g_return_if_fail (object != NULL);
+        g_return_if_fail (GTK_IS_HTML (object));
+
+        html = (GtkHtml *)object;
+
+        gtk_object_unref (GTK_OBJECT (html->hadj));
+        gtk_object_unref (GTK_OBJECT (html->vadj));
+
+        hbits = html->html_bits;
+        
+        while (hbits) {
+                hb = (GtkHtmlBit *)hbits->data;
+                if (hb->fore)
+                        gdk_color_free(hb->fore);
+                if (hb->back)
+                        gdk_color_free(hb->back);
+                if (hb->text)
+                        g_free(hb->text);
+                if (hb->url)
+                        g_free(hb->url);
+                if (hb->pm)
+                        gdk_pixmap_unref(hb->pm);
+		
+                g_free(hb);
+                hbits = hbits->next;
+        }
+        if (html->html_bits)
+                g_list_free(html->html_bits);
+
+        if (html->urls)
+                g_list_free(html->urls);
+
+        if (html->selected_text)
+                g_free(html->selected_text);
+
+        if (html->gc)
+                gdk_gc_destroy(html->gc);
+
+        if (html->bg_gc)
+                gdk_gc_destroy(html->bg_gc);
+
+	if (html->tooltip_window)
+		gtk_widget_destroy(html->tooltip_window);
+        
+        GTK_OBJECT_CLASS(parent_class)->finalize (object);
+}
+
+static void gtk_html_realize (GtkWidget *widget)
+{
+        GtkHtml *html;
+        GdkWindowAttr attributes;
+        gint attributes_mask;
+  
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+  
+        html = GTK_HTML (widget);
+        GTK_WIDGET_SET_FLAGS (html, GTK_REALIZED);
+  
+        attributes.window_type = GDK_WINDOW_CHILD;
+        attributes.x = widget->allocation.x;
+        attributes.y = widget->allocation.y;
+        attributes.width = widget->allocation.width;
+        attributes.height = widget->allocation.height;
+        attributes.wclass = GDK_INPUT_OUTPUT;
+        attributes.visual = gtk_widget_get_visual (widget);
+        attributes.colormap = gtk_widget_get_colormap (widget);
+        attributes.event_mask = gtk_widget_get_events (widget);
+        attributes.event_mask |= (GDK_EXPOSURE_MASK |
+                                  GDK_BUTTON_PRESS_MASK |
+                                  GDK_BUTTON_RELEASE_MASK |
+                                  GDK_BUTTON_MOTION_MASK |
+                                  GDK_ENTER_NOTIFY_MASK |
+                                  GDK_LEAVE_NOTIFY_MASK |
+                                  GDK_POINTER_MOTION_MASK |
+                                  GDK_POINTER_MOTION_HINT_MASK |
+                                  GDK_VISIBILITY_NOTIFY_MASK |
+                                  GDK_KEY_PRESS_MASK);
+        
+        attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
+  
+        widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, attributes_mask);
+        gdk_window_set_user_data (widget->window, html);
+  
+        attributes.x = (widget->style->klass->xthickness + BORDER_WIDTH);
+        attributes.y = (widget->style->klass->ythickness + BORDER_WIDTH);
+        attributes.width = MAX (1, (gint)widget->allocation.width - (gint)attributes.x * 2);
+        attributes.height = MAX (1, (gint)widget->allocation.height - (gint)attributes.y * 2);
+  
+        html->html_area = gdk_window_new (widget->window, &attributes, attributes_mask);
+        gdk_window_set_user_data (html->html_area, html);
+  
+        widget->style = gtk_style_attach (widget->style, widget->window);
+  
+        /* Can't call gtk_style_set_background here because it's handled specially */
+        gdk_window_set_background (widget->window, &widget->style->base[GTK_STATE_NORMAL]);
+        gdk_window_set_background (html->html_area, &widget->style->base[GTK_STATE_NORMAL]);
+
+        if (widget->style->bg_pixmap[GTK_STATE_NORMAL])
+                html->bg_gc = create_bg_gc(html);
+  
+        html->gc = gdk_gc_new (html->html_area);
+        gdk_gc_set_exposures (html->gc, TRUE);
+        gdk_gc_set_foreground (html->gc, &widget->style->text[GTK_STATE_NORMAL]);
+  
+        gdk_window_show(html->html_area);
+
+}
+
+static void gtk_html_style_set(GtkWidget      *widget,
+                               GtkStyle       *previous_style)
+{
+        GtkHtml *html;
+  
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+  
+        html = GTK_HTML (widget);
+        if (GTK_WIDGET_REALIZED (widget)) {
+                gdk_window_set_background (widget->window, &widget->style->base[GTK_STATE_NORMAL]);
+                gdk_window_set_background (html->html_area, &widget->style->base[GTK_STATE_NORMAL]);
+      
+                if (html->bg_gc) {
+                        gdk_gc_destroy (html->bg_gc);
+                        html->bg_gc = NULL;
+                }
+
+                if (widget->style->bg_pixmap[GTK_STATE_NORMAL]) {
+                        html->bg_gc = create_bg_gc(html);
+                }
+
+        }
+}
+
+static void gtk_html_unrealize (GtkWidget *widget)
+{
+        GtkHtml *html;
+  
+        g_return_if_fail (widget != NULL);
+        g_return_if_fail (GTK_IS_HTML (widget));
+  
+        html = GTK_HTML (widget);
+
+        gdk_window_set_user_data (html->html_area, NULL);
+        gdk_window_destroy (html->html_area);
+        html->html_area = NULL;
+  
+        gdk_gc_destroy (html->gc);
+        html->gc = NULL;
+
+        if (html->bg_gc)
+        {
+                gdk_gc_destroy (html->bg_gc);
+                html->bg_gc = NULL;
+        }
+
+        if (GTK_WIDGET_CLASS (parent_class)->unrealize)
+                (* GTK_WIDGET_CLASS (parent_class)->unrealize) (widget);
+}
+
+
+
+
+
+static void gtk_html_add_pixmap(GtkHtml *html,
+                                GdkPixmap *pm,
+                                int fit)
+{
+        GtkHtmlBit *last_hb;
+        GtkHtmlBit *hb = g_new0(GtkHtmlBit, 1);
+        GdkWindowPrivate *private = (GdkWindowPrivate *)pm;
+
+        last_hb = (GtkHtmlBit *)g_list_last(html->html_bits)->data;
+
+	hb->fit = fit;
+	hb->x = html->current_x;
+	hb->y = html->current_y;
+        if (fit)
+		hb->height = last_hb->height;
+	else
+		hb->height = private->height;
+	hb->type = HTML_BIT_PIXMAP;
+	hb->width = private->width;
+	hb->text = NULL;
+	hb->url = NULL;
+	hb->fore = NULL;
+	hb->back = NULL;
+	hb->font = NULL;
+        hb->uline = 0;
+        hb->strike = 0;
+	hb->was_selected = 0;
+        hb->newline = 0;
+        hb->pm = pm;
+
+        if (html->current_x == BORDER_WIDTH) {
+		html->current_y += hb->height;
+		hb->y += hb->height;
+	}
+
+
+        html->current_x += hb->width;
+	
+	gtk_html_draw_bit(html, hb, 1);
+
+	html->html_bits = g_list_append(html->html_bits, hb);
+
+
+}
+
+static void gtk_html_add_seperator(GtkHtml *html)
+{
+        GtkHtmlBit *hb = g_new0(GtkHtmlBit, 1);
+        gint width, height;
+	
+	html->current_x = 0;
+	html->current_y += 5;
+
+        gdk_window_get_size(html->html_area, &width, &height);
+        
+	hb->x = html->current_x;
+	hb->y = html->current_y;
+	hb->height = 5;
+	hb->type = HTML_BIT_SEP;
+	hb->width = width - GTK_SCROLLED_WINDOW(GTK_WIDGET(html)->parent)->vscrollbar->allocation.width - 10;
+	hb->text = NULL;
+	hb->url = NULL;
+	hb->fore = NULL;
+	hb->back = NULL;
+	hb->font = NULL;
+        hb->uline = 0;
+        hb->strike = 0;
+	hb->was_selected = 0;
+        hb->newline = 0;
+        hb->pm = NULL;
+
+	gtk_html_draw_bit(html, hb, 1);
+
+	html->html_bits = g_list_append(html->html_bits, hb);
+
+}
+
+
+static void gtk_html_add_text (GtkHtml *html,
+                               GdkFont *cfont,
+                               GdkColor *fore,
+                               GdkColor *back,
+                               char *chars,
+                               gint length,
+                               gint uline,
+                               gint strike,
+                               char *url)
+{
+        char *nextline = NULL, *c, *text, *tmp;
+        GdkGC *gc;
+        int nl = 0, nl2 = 0;
+        int maxwidth;
+        gint lb;
+        GList *hbits;
+        int num = 0, i, height;
+        GtkHtmlBit *hb;
+        gint hwidth, hheight;
+
+        if (length == 1 && chars[0] == '\n') {
+                GtkHtmlBit *h;
+                hbits = g_list_last(html->html_bits);
+                if (!hbits)
+                        return;
+                /* I realize this loses a \n sometimes
+                 * if it's the first thing in the widget.
+                 * so fucking what. */
+
+                h = (GtkHtmlBit *)hbits->data;
+                h->newline++;
+                if (html->current_x > 0)
+                        html->current_x = 0;
+                else
+                        html->current_y += gdk_string_height(cfont, "yG") + 2;
+                return;
+        }
+
+
+	
+        c = text = g_malloc(length + 2);
+        strncpy(text, chars, length);
+        text[length] = 0;
+
+
+        gc = html->gc;
+
+        if (gc == NULL)
+                gc = html->gc = gdk_gc_new(html->html_area);
+
+        gdk_gc_set_font(gc, cfont);
+
+
+        while(*c) {
+                if (*c == '\n') {
+                        if (*(c+1) == '\0') {
+                                nl = 1;
+                                length--;
+                                c[0] = '\0';
+                                break;
+                        }
+                        if (*c) {
+                                gtk_html_add_text(html, cfont, fore, back, text, num + 1, uline, strike, url);
+                                tmp = text;
+                                length -= (num+1);
+                                text = g_malloc(length+2);
+                                strncpy(text, (c+1), length);
+                                text[length] = 0;
+                                c = text;
+                                num = 0;
+                                g_free(tmp);
+				continue;
+                        }
+                }
+
+                num++;
+                c++;
+        }
+
+        /* Note, yG is chosen because G is damn high, and y is damn low, */
+        /* it should be just fine. :) */
+
+        gdk_window_get_size(html->html_area, &hwidth, &hheight);
+        
+        num = strlen(text);
+
+        while(GTK_WIDGET(html)->allocation.width < 20) {
+		while(gtk_events_pending())
+			gtk_main_iteration();
+	}
+
+        maxwidth = (hwidth - html->current_x - 8);
+                    /*HTK_SCROLLED_WINDOW(GTK_WIDGET(layout)->parent)->vscrollbar->allocation.width) - 8; */
+
+        while(gdk_text_measure(cfont, text, num) > maxwidth) {
+                if (num > 1)
+                        num--;
+                else {
+
+                        html->current_x = 0;
+                        if (nl) {
+				text[length] = '\n';
+                                length++;
+			}
+			gtk_html_add_text(html, cfont, fore, back, text, length, uline, strike, url);
+			g_free(text);
+			return;
+		}
+
+	}
+
+	height = gdk_string_height(cfont, "yG") + 2;
+
+        
+        if ((int)(html->vadj->upper - html->current_y) < (int)(height * 2)) {
+                int val;
+                val = (height * 2) + html->current_y;
+                html->vadj->upper = val;
+                adjust_adj(html, html->vadj);
+	}
+
+	
+	if (html->current_x == 0) {
+                html->current_y += height;
+                gdk_text_extents(cfont, text, 1, &lb, NULL, NULL, NULL, NULL);
+                html->current_x += (2 - lb);
+	} else if ((hbits = g_list_last(html->html_bits)) != NULL) {
+	        int diff, y;
+		hb = (GtkHtmlBit *)hbits->data;
+		if (height > hb->height) {
+			diff = height - hb->height;
+			y = hb->y;
+			html->current_y += diff;
+			while(hbits) {
+				hb = (GtkHtmlBit *)hbits->data;
+				if (hb->y != y)
+                                        break;
+                                hb->height = height;
+                                hb->y += diff;
+
+                                hbits = hbits->prev;
+			}
+		}
+	}
+
+
+
+
+	if (num != strlen(text)) {
+		/* This is kinda cheesy but it may make things
+		 * much better lookin */
+		for (i=2; i<15; i++) {
+			if ((num - i) < 0) {
+				html->current_x = 0;
+				gtk_html_add_text(html, cfont, fore, back, text, strlen(text), uline, strike, url);
+                                return;
+			} else if (text[num - i] == ' ') {
+				num = num - (i-1);
+				nl2 = 1;
+				break;
+			}
+		}
+
+		nextline = g_malloc(length - num + 2);
+		strncpy(nextline, (char *)(text + num), length - num);
+		nextline[length - num] = 0;
+		if (nl) {
+			nextline[length - num] = '\n';
+			nextline[length - num + 1] = 0;
+			nl = 0;
+		}
+
+
+		text[num] = 0;
+	}
+
+
+	if (url != NULL)
+		fore = get_color(3355647, gdk_window_get_colormap(html->html_area));
+
+
+	hb = g_new0(GtkHtmlBit, 1);
+
+	hb->text = g_strdup(text);
+
+        if (fore)
+		hb->fore = gdk_color_copy(fore);
+	else
+		hb->fore = NULL;
+
+	if (back)
+		hb->back = gdk_color_copy(back);
+	else
+		hb->back = NULL;
+	hb->font = cfont;
+	hb->uline = uline;
+	hb->strike = strike;
+	hb->height = height;
+	gdk_text_extents(cfont, text, num, &lb, NULL, &hb->width, NULL, NULL);
+	hb->x = html->current_x;
+	hb->y = html->current_y;
+        hb->type = HTML_BIT_TEXT;
+        hb->pm = NULL;
+	if (url != NULL) {
+		uline = 1;
+		hb->uline = 1;
+		hb->url = g_strdup(url);
+	} else {
+		hb->url = NULL;
+	}
+	html->current_x += hb->width;
+
+	html->html_bits = g_list_append(html->html_bits, hb);
+        if (url != NULL) {
+                html->urls = g_list_append(html->urls, hb);
+        }
+
+
+
+        gtk_html_draw_bit(html, hb, 1);
+
+        if (nl || nl2) {
+                if (nl)
+                        hb->newline = 1;
+                html->current_x = 0;
+        } else
+                hb->newline = 0;
+
+
+        if (nextline != NULL) {
+                gtk_html_add_text(html, cfont, fore, back, nextline, strlen(nextline), uline, strike, url);
+                g_free(nextline);
+        }
+
+        g_free(text);
+
+
+}
+
+
+void gtk_html_append_text (GtkHtml *html,
+			   char *text,
+			   gint options)
+{
+        GdkColormap *map;
+	GdkFont *cfont;
+        GdkRectangle area;
+        char ws[BUF_LONG], tag[BUF_LONG], *c, *url = NULL;
+        gint intag=0,wpos=0, tpos=0, colorv, bold=0, italic=0, fixed=0, uline=0, strike=0, title=0;
+        gint height;
+        struct font_state *current, *tmp;
+        struct font_state def_state = { 3, 0, 0, NULL, NULL, NULL };
+
+        current = &def_state;
+        map = gdk_window_get_colormap(html->html_area);
+        cfont = getfont(bold, italic, fixed, current->size);
+	c = text;
+
+
+        while(*c) {
+                if (*c == '<') {
+                        if (!intag) {
+                                ws[wpos]=0;
+                                if (wpos) {
+                                        if (title) {
+                                                if (html->title)
+                                                        g_free(html->title);
+                                                html->title = g_strdup(ws);
+                                        } else
+                                                gtk_html_add_text(html, cfont, current->color, current->bgcol, ws, strlen(ws), uline, strike, url);
+                                }
+                                wpos=0;
+                                intag=1;
+                        } else {
+                                /* Assuming you NEVER have nested tags
+                                 * (and I mean <tag <tag>> by this, not
+                                 * <tag><tag2></tag2><tag>..*/
+                                tag[tpos] = 0;
+				gtk_html_add_text(html, cfont, current->color, current->bgcol, "<", 1, 0, 0, NULL);
+				gtk_html_add_text(html, cfont, current->color, current->bgcol, tag, strlen(tag), 0, 0, NULL);
+				tpos = 0;
+ 
+				tag[0]=*c;
+			}
+		} else if (*c == '>') {
+			if (intag) {
+				tag[tpos]=0;
+				if (!strcasecmp(tag, "B"))
+					bold = 1;
+				else if (!strcasecmp(tag, "STRIKE"))
+					strike = 1;
+				else if (!strcasecmp(tag, "I"))
+					italic = 1;
+				else if (!strcasecmp(tag, "U"))
+					uline = 1;
+				else if (!strcasecmp(tag, "PRE"))
+					fixed = 1;
+				else if (!strcasecmp(tag, "HR"))
+					gtk_html_add_seperator(html);
+				else if (!strcasecmp(tag, "/B"))
+					bold = 0;
+				else if (!strcasecmp(tag, "/STRIKE"))
+					strike = 0;
+				else if (!strcasecmp(tag, "/I"))
+					italic = 0;
+				else if (!strcasecmp(tag, "/U"))
+					uline = 0;
+				else if (!strcasecmp(tag, "/PRE"))
+					fixed = 0;
+				else if (!strcasecmp(tag, "TITLE"))
+					title = 1;
+				else if (!strcasecmp(tag, "/TITLE"))
+					title = 0;
+				else if (!strncasecmp(tag, "IMG", 3)) {
+					
+				} else if (!strcasecmp(tag, "H3")) {
+					current = push_state(current);
+					current->size = 4;
+                                } else if (!strcasecmp(tag, "/H3")) {
+					gtk_html_add_text(html, cfont, current->color, current->bgcol, "\n", 1, 0, 0, NULL);
+
+					if (current->next) {
+						if (current->ownbg)
+							g_free(current->bgcol);
+						if (current->owncolor)
+							g_free(current->color);
+                                                tmp=current;
+						current=current->next;
+                                                g_free(tmp);
+					}
+				} else if (!strcasecmp(tag, "TABLE")) {
+                                } else if (!strcasecmp(tag, "/TABLE")) {
+				} else if (!strcasecmp(tag, "TR")) {
+				} else if (!strcasecmp(tag, "/TR")) {
+				} else if (!strcasecmp(tag, "/TD")) {
+				} else if (!strcasecmp(tag, "TD")){
+					gtk_html_add_text(html, cfont, current->color, current->bgcol, "  ", 2, 0, 0, NULL);
+				} else if (!strncasecmp(tag, "A ", 2)) {
+					char *d;
+					char *temp = d = g_strdup(tag);
+					int flag = 0;
+					strtok(tag," ");
+                                        while((d=strtok(NULL," "))) {
+						if (strlen(d) < 7)
+							break;
+						if (!strncasecmp(d, "HREF=\"", strlen("HREF=\""))) {
+							d+= strlen("HREF=\"");
+							d[strlen(d) - 1] = 0;
+							url = g_malloc(strlen(d) + 1);
+							strcpy(url, d);
+							flag = 1;
+                                                }
+                                        }
+					g_free(temp);
+					if (!flag) {
+						gtk_html_add_text(html, cfont, current->color, current->bgcol, "<", 1, 0, 0, NULL);
+						gtk_html_add_text(html, cfont, current->color, current->bgcol, tag, strlen(tag), 0, 0, NULL);
+						gtk_html_add_text(html, cfont, current->color, current->bgcol, ">", 1, 0, 0, NULL);
+					}
+				} else if (!strcasecmp(tag, "/A")) {
+					if (url) {
+						g_free(url);
+						url = NULL;
+					}
+				} else if (!strncasecmp(tag,"FONT", strlen("FONT"))) {
+                                        char *d;
+					/* Push a new state onto the stack, based on the old state */
+                                        current = push_state(current);
+					strtok(tag," ");
+                                        while((d=strtok(NULL," "))) {
+                                                if (!strncasecmp(d,"COLOR=",strlen("COLOR="))) {
+                                                        d+=strlen("COLOR=");
+                                                        if (*d == '\"') d++;
+                                                        if (*d == '#') d++;
+                                                        if (d[strlen(d) - 1] == '\"')
+                                                                d[strlen(d) - 1] = 0;
+                                                        if (sscanf(d, "%x", &colorv) && !(options & HTML_OPTION_NO_COLOURS)) {
+								current->color = get_color(colorv, map);
+								current->owncolor = 1;
+							} else {
+								sprintf(debug_buff,"didn't find color in '%s'\n",d);
+								debug_print(debug_buff);
+							}
+						} else 
+						if (!strncasecmp(d,"BACK=",strlen("BACK="))) {
+							d+=strlen("BACK=");
+							if (*d == '\"') d++;
+							if (*d == '#') d++;
+							if (d[strlen(d) - 1] == '\"')
+								d[strlen(d) - 1] = 0;
+							if (sscanf(d, "%x", &colorv) && !(options & HTML_OPTION_NO_COLOURS)) {
+								current->bgcol = get_color(colorv, map);
+								current->ownbg = 1;
+							} else {
+								sprintf(debug_buff,"didn't find color in '%s'\n",d);
+								debug_print(debug_buff);
+							}
+						} else if (!strncasecmp(d,"SIZE=",strlen("SIZE="))) {
+                                                        d+=strlen("SIZE=");
+                                                        if (*d == '\"') d++;
+                                                        if (*d == '+') d++;
+                                                        if (sscanf(d, "%d", &colorv)) {
+                                                                current->size = colorv;
+                                                        } else {
+								sprintf(debug_buff,"didn't find size in '%s'\n",d);
+								debug_print(debug_buff);
+                                                        }
+                                                } else if (strncasecmp(d,"PTSIZE=", strlen("PTSIZE="))) {
+                                                }
+                                        }
+                                } else if (!strncasecmp(tag,"BODY BGCOLOR", strlen("BODY BGCOLOR"))) {
+
+                                        /* Ditch trailing \" */
+                                        tag[strlen(tag)-1]=0;
+                                        if (sscanf(tag + strlen("BODY BGCOLOR=\"#"), "%x", &colorv) && !(options & HTML_OPTION_NO_COLOURS)) {
+                                                current->bgcol = get_color(colorv, map);
+                                                current->ownbg = 1;
+                                        }
+                                } else if (!strncasecmp(tag, "/FONT", strlen("/FONT"))) {
+                                        /* Pop a font state off the list if possible, freeing
+                                         any resources it used */
+                                        if (current->next) {
+                                                if (current->ownbg)
+                                                        g_free(current->bgcol);
+                                                if (current->owncolor)
+                                                        g_free(current->color);
+                                                tmp=current;
+                                                current=current->next;
+                                                g_free(tmp);
+					}
+
+				} else if (!strcasecmp(tag, "/BODY")) {
+					if (current->next) {
+						if (current->ownbg)
+							g_free(current->bgcol);
+						if (current->owncolor)
+							g_free(current->color);
+						tmp=current;
+						current=current->next;
+						g_free(tmp);
+					}	/* tags we ignore below */
+				} else if (!strncasecmp(tag, "BR", 2)) {
+					gtk_html_add_text(html, cfont, current->color, current->bgcol, "\n", 1, 0, 0, NULL);
+				} else if (strncasecmp(tag, "HTML", 4) && strncasecmp(tag, "/HTML", 5) &&
+					   strncasecmp(tag, "BODY", 4) && strncasecmp(tag, "/BODY", 5) &&
+					   strncasecmp(tag, "P", 1) && strncasecmp(tag, "/P", 2) &&
+					   strncasecmp(tag, "HEAD", 4) && strncasecmp(tag, "/HEAD", 5)) {
+					if (tpos) {
+						gtk_html_add_text(html, cfont, current->color, current->bgcol, "<", 1, 0, 0, NULL);
+						gtk_html_add_text(html, cfont, current->color, current->bgcol, tag, strlen(tag), 0, 0, NULL);
+						gtk_html_add_text(html, cfont, current->color, current->bgcol, ">", 1, 0, 0, NULL);
+
+					}
+				}
+				cfont = getfont(bold,italic,fixed,current->size);
+				tpos=0;
+				intag = 0;
+			} else {
+				ws[wpos++]=*c;
+			}
+		} else if (!intag && *c == '&') {
+                if (!strncasecmp(c, "&amp;", 5)) {
+				ws[wpos++] = '&';
+				c+=4;
+			} else if (!strncasecmp(c, "&lt;", 4)) {
+				ws[wpos++] = '<';
+				c+=3;
+			} else if (!strncasecmp(c, "&gt;", 4)) {
+				ws[wpos++] = '>';
+                                c+=3;
+                        } else if (!strncasecmp(c, "&nbsp;", 6)) {
+				ws[wpos++] = ' ';
+				c+=5;
+			} else {
+				ws[wpos++] = *c;
+			}
+                } else {
+			if (intag) {
+				tag[tpos++]=*c;
+			} else {
+				ws[wpos++]=*c;
+			}
+		}
+		c++;
+	}
+	while(current->next) {
+		if (current->ownbg)
+			g_free(current->bgcol);
+		if (current->owncolor)
+			g_free(current->color);
+		tmp = current;
+		current = current->next;
+		g_free(tmp);
+	}
+	ws[wpos]=0;
+	tag[tpos]=0;
+	if (wpos) {
+		gtk_html_add_text(html, cfont, current->color, current->bgcol, ws, strlen(ws), uline, strike, url);
+	}
+	if (tpos) {
+		gtk_html_add_text(html, cfont, current->color, current->bgcol, "<", 1, 0, 0, NULL);
+		gtk_html_add_text(html, cfont, current->color, current->bgcol, tag, strlen(tag), 0, 0, NULL);
+		gtk_html_add_text(html, cfont, current->color, current->bgcol, ">", 1, 0, 0, NULL);
+	}
+
+
+	
+        gdk_window_get_size(html->html_area, NULL, &height);
+        area.height = height;
+	gtk_adjustment_set_value(html->vadj, html->vadj->upper - area.height);
+
+        return;
+}
+
+
+static void adjust_adj (GtkHtml *html,
+                        GtkAdjustment *adj)
+{
+        gint height;
+  
+        gdk_window_get_size (html->html_area, NULL, &height);
+  
+        adj->step_increment = MIN (adj->upper, (float) SCROLL_PIXELS);
+        adj->page_increment = MIN (adj->upper, height - (float) KEY_SCROLL_PIXELS);
+        adj->page_size      = MIN (adj->upper, height);
+        adj->value          = MIN (adj->value, adj->upper - adj->page_size);
+        adj->value          = MAX (adj->value, 0.0);
+  
+        gtk_signal_emit_by_name (GTK_OBJECT (adj), "changed");
+}
+
+
+static void scroll_down (GtkHtml* html,
+                         gint diff0)
+{
+        GdkRectangle rect;
+        gint width, height;
+  
+        html->yoffset += diff0;
+  
+	gdk_window_get_size (html->html_area, &width, &height);
+
+	if (html->transparent) {
+		rect.x = 0;
+		rect.y = 0;
+		rect.width = width;
+		rect.height = height;
+	} else {
+		
+
+		if (height > diff0 && !html->transparent)
+			gdk_draw_pixmap (html->html_area,
+					 html->gc,
+					 html->html_area,
+					 0,
+					 diff0,
+					 0,
+					 0,
+					 width,
+					 height - diff0);
+
+		rect.x      = 0;
+		rect.y      = MAX (0, height - diff0);
+		rect.width  = width;
+		rect.height = MIN (height, diff0);
+	}
+		
+        expose_html (html, &rect, FALSE);
+        gtk_html_draw_focus ( (GtkWidget *) html);
+
+}
+
+static void scroll_up (GtkHtml* html,
+                         gint diff0)
+{
+        GdkRectangle rect;
+        gint width, height;
+  
+	html->yoffset -= diff0;
+
+
+	gdk_window_get_size (html->html_area, &width, &height);
+
+	if (html->transparent) {
+		rect.x = 0;
+		rect.y = 0;
+		rect.width = width;
+		rect.height = height;
+	} else {
+  
+		if (height > diff0)
+			gdk_draw_pixmap (html->html_area,
+					 html->gc,
+					 html->html_area,
+					 0,
+					 0,
+					 0,
+					 diff0,
+					 width,
+					 height - diff0);
+
+		rect.x      = 0;
+		rect.y      = 0;
+		rect.width  = width;
+		rect.height = MIN (height, diff0);
+	}
+
+        expose_html (html, &rect, FALSE);
+        gtk_html_draw_focus ( (GtkWidget *) html);
+
+}
+
+
+
+static void gtk_html_adjustment (GtkAdjustment *adjustment,
+                                 GtkHtml       *html)
+{
+        g_return_if_fail (adjustment != NULL);
+        g_return_if_fail (GTK_IS_ADJUSTMENT (adjustment));
+        g_return_if_fail (html != NULL);
+        g_return_if_fail (GTK_IS_HTML (html));
+
+        /* Just ignore it if we haven't been size-allocated and realized yet */
+        if (html->html_area == NULL)
+                return;
+  
+        if (adjustment == html->hadj) {
+                g_warning ("horizontal scrolling not implemented");
+        } else {
+                gint diff = ((gint)adjustment->value) - html->last_ver_value;
+      
+                if (diff != 0) {
+                        /*undraw_cursor (text, FALSE);*/
+          
+                        if (diff > 0) {
+                                scroll_down (html, diff);
+                        } else {/* if (diff < 0) */
+                                scroll_up (html, -diff);
+                        }
+                        /*draw_cursor (text, FALSE); */
+
+                        html->last_ver_value = adjustment->value;
+                }
+        }
+}
+ 
+static gint gtk_html_visibility_notify (GtkWidget *widget,
+                                        GdkEventVisibility *event)
+{
+	GtkHtml *html;
+	GdkRectangle rect;
+        gint width, height;
+
+	g_return_val_if_fail (widget != NULL, FALSE);
+        g_return_val_if_fail (GTK_IS_HTML (widget), FALSE);
+
+        html = GTK_HTML(widget);
+        
+        if (GTK_WIDGET_REALIZED (widget) && html->transparent) {
+                gdk_window_get_size (html->html_area, &width, &height);
+                rect.x = 0;
+                rect.y = 0;
+                rect.width = width;
+                rect.height = height;
+                expose_html (html, &rect, FALSE);
+                gtk_html_draw_focus ( (GtkWidget *) html);
+        } else {
+	}
+
+
+        return FALSE;
+}
+
+
+
+static void gtk_html_disconnect (GtkAdjustment *adjustment,
+                                 GtkHtml       *html)
+{
+        g_return_if_fail (adjustment != NULL);
+        g_return_if_fail (GTK_IS_ADJUSTMENT (adjustment));
+        g_return_if_fail (html != NULL);
+        g_return_if_fail (GTK_IS_HTML (html));
+
+        if (adjustment == html->hadj)
+                gtk_html_set_adjustments (html, NULL, html->vadj);
+        if (adjustment == html->vadj)
+                gtk_html_set_adjustments (html, html->hadj, NULL);
+}
+
+static void move_cursor_ver(GtkHtml *html, int count)
+{
+	GList *hbits = g_list_find(html->html_bits, html->cursor_hb);
+	GtkHtmlBit *hb = NULL, *hb2 = NULL;
+	gint y;
+        gint len, len2 = 0;
+	
+	undraw_cursor(html);
+
+	if (!html->html_bits)
+		return;
+	
+	if (!html->cursor_hb)
+		html->cursor_hb = (GtkHtmlBit *)html->html_bits->data;
+
+	hb = html->cursor_hb;
+
+	len = html->cursor_pos;
+	hbits = hbits->prev;
+	while(hbits) {
+		hb2 = (GtkHtmlBit *)hbits->data;
+
+		if (hb2->y != hb->y)
+			break;
+
+                len += strlen(hb2->text);
+		
+		hbits = hbits->prev;
+	}
+
+        hbits = g_list_find(html->html_bits, html->cursor_hb);
+	
+	if (count < 0) {
+		while(hbits) {
+			hb2 = (GtkHtmlBit *)hbits->data;
+
+			if (hb2->y != hb->y)
+				break;
+			
+			hbits = hbits->prev;
+		}
+		if (!hbits) {
+			draw_cursor(html);
+			return;
+		}
+		y = hb2->y;
+		hb = hb2;
+		while(hbits) {
+			hb2 = (GtkHtmlBit *)hbits->data;
+
+			if (hb2->y != y)
+				break;
+
+			hb = hb2;
+			
+			hbits = hbits->prev;
+		}
+		hbits = g_list_find(html->html_bits, hb);
+		while(hbits) {
+			hb2 = (GtkHtmlBit *)hbits->data;
+
+			if (hb->y != hb2->y) {
+				html->cursor_hb = hb;
+				html->cursor_pos = strlen(hb->text);
+                                break;
+			}
+
+
+			if (len < len2 + strlen(hb2->text)) {
+				html->cursor_hb = hb2;
+				html->cursor_pos = len - len2;
+				break;
+			}
+
+			len2 += strlen(hb2->text);
+
+			hb = hb2;
+
+                        hbits = hbits->next;
+		}
+	} else {
+		while(hbits) {
+			hb2 = (GtkHtmlBit *)hbits->data;
+
+			if (hb2->y != hb->y)
+				break;
+			
+			hbits = hbits->next;
+		}
+		if (!hbits) {
+			draw_cursor(html);
+			return;
+		}
+		hb = hb2;
+		while(hbits) {
+			hb2 = (GtkHtmlBit *)hbits->data;
+
+			if (hb->y != hb2->y) {
+				html->cursor_hb = hb;
+				html->cursor_pos = strlen(hb->text);
+                                break;
+			}
+
+
+			if (len < len2 + strlen(hb2->text)) {
+				html->cursor_hb = hb2;
+				html->cursor_pos = len - len2;
+				break;
+			}
+
+			len2 += strlen(hb2->text);
+
+			hb = hb2;
+
+                        hbits = hbits->next;
+		}
+	}
+
+	draw_cursor(html);
+
+}
+
+static void move_cursor_hor(GtkHtml *html, int count)
+{
+	GList *hbits = g_list_find(html->html_bits, html->cursor_hb);
+	GtkHtmlBit *hb, *hb2;
+
+	undraw_cursor(html);
+
+	if (!html->html_bits)
+		return;
+	
+	if (!html->cursor_hb)
+		html->cursor_hb = (GtkHtmlBit *)html->html_bits->data;
+
+	html->cursor_pos+=count;
+	
+	if (html->cursor_pos < 0) {
+		if (hbits->prev) {
+			gint diff;
+			hb = html->cursor_hb;
+			hb2 = (GtkHtmlBit *)hbits->prev->data;
+			diff = html->cursor_pos + strlen(hb2->text) + 1;
+			if (hb->y == hb2->y)
+				--diff;
+			
+			html->cursor_pos = diff;
+			
+                        html->cursor_hb = (GtkHtmlBit *)hbits->prev->data;
+		} else {
+			html->cursor_pos = 0;
+		}
+	} else if (html->cursor_pos > strlen(html->cursor_hb->text)) {
+		if (hbits->next) {
+			gint diff;
+			hb = html->cursor_hb;
+			hb2 = (GtkHtmlBit *)hbits->next->data;
+
+			diff = html->cursor_pos - strlen(html->cursor_hb->text) - 1;
+			if (hb->y == hb2->y)
+                                ++diff;
+			html->cursor_pos = diff;
+                        html->cursor_hb = (GtkHtmlBit *)hbits->next->data;
+		} else {
+			html->cursor_pos = strlen(html->cursor_hb->text);
+		}
+
+	}
+
+	draw_cursor(html);
+}
+
+static void move_beginning_of_line(GtkHtml *html)
+{
+	GList *hbits = g_list_find(html->html_bits, html->cursor_hb);
+	GtkHtmlBit *hb = NULL;
+        gint y;
+	
+	undraw_cursor(html);
+
+	if (!html->html_bits)
+		return;
+
+	if (!html->cursor_hb)
+		html->cursor_hb = (GtkHtmlBit *)html->html_bits->data;
+
+	y = html->cursor_hb->y;
+	
+	while(hbits) {
+		hb = (GtkHtmlBit *)hbits->data;
+
+		if (y != hb->y) {
+			hb = (GtkHtmlBit *)hbits->next->data;
+			break;
+		}
+		
+		hbits = hbits->prev;
+	}
+	if (!hbits)
+		html->cursor_hb = (GtkHtmlBit*)html->html_bits->data;
+	else
+		html->cursor_hb = hb;
+
+	html->cursor_pos = 0;
+
+
+	draw_cursor(html);
+
+
+}
+
+static void move_end_of_line(GtkHtml *html)
+{
+	GList *hbits = g_list_find(html->html_bits, html->cursor_hb);
+	GtkHtmlBit *hb = NULL;
+        gint y;
+	
+	undraw_cursor(html);
+
+	if (!html->html_bits)
+		return;
+
+	if (!html->cursor_hb)
+		html->cursor_hb = (GtkHtmlBit *)html->html_bits->data;
+
+	y = html->cursor_hb->y;
+	
+	while(hbits) {
+		hb = (GtkHtmlBit *)hbits->data;
+
+		if (y != hb->y) {
+			hb = (GtkHtmlBit *)hbits->prev->data;
+			break;
+		}
+		
+		hbits = hbits->next;
+	}
+	if (!hbits)
+		html->cursor_hb = (GtkHtmlBit*)g_list_last(html->html_bits)->data;
+	else
+		html->cursor_hb = hb;
+
+	html->cursor_pos = strlen(html->cursor_hb->text);
+
+
+	draw_cursor(html);
+
+
+}
+
+
+
+static gint
+gtk_html_key_press (GtkWidget   *widget,
+                    GdkEventKey *event) 
+{
+	GtkHtml *html;
+	gchar key;
+	gint return_val;
+  
+	g_return_val_if_fail (widget != NULL, FALSE);
+	g_return_val_if_fail (GTK_IS_HTML (widget), FALSE);
+	g_return_val_if_fail (event != NULL, FALSE);
+  
+	return_val = FALSE;
+  
+	html = GTK_HTML (widget);
+  
+	key = event->keyval;
+	return_val = TRUE;
+
+
+	if (html->editable == FALSE) {
+                /*
+		switch (event->keyval) {
+		case GDK_Home:
+			if (event->state & GDK_CONTROL_MASK)
+				scroll_int (text, -text->vadj->value);
+			else
+				return_val = FALSE;
+			break;
+		case GDK_End:
+			if (event->state & GDK_CONTROL_MASK)
+				scroll_int (text, +text->vadj->upper);
+			else
+				return_val = FALSE;
+			break;
+		case GDK_Page_Up:   scroll_int (text, -text->vadj->page_increment); break;
+		case GDK_Page_Down: scroll_int (text, +text->vadj->page_increment); break;
+		case GDK_Up:        scroll_int (text, -KEY_SCROLL_PIXELS); break;
+		case GDK_Down:      scroll_int (text, +KEY_SCROLL_PIXELS); break;
+		case GDK_Return:
+			if (event->state & GDK_CONTROL_MASK)
+				gtk_signal_emit_by_name (GTK_OBJECT (text), "activate");
+			else
+				return_val = FALSE;
+			break;
+		default:
+			return_val = FALSE;
+			break;
+		}
+		*/
+	} else {
+      
+		switch (event->keyval) {
+		case GDK_Home:
+			move_beginning_of_line (html);
+			break;
+                case GDK_End:
+			move_end_of_line (html);
+			break;
+			/*
+		case GDK_Page_Up:
+			move_cursor_page_ver (html, -1);
+			break;
+		case GDK_Page_Down:
+			move_cursor_page_ver (html, +1);
+			break;*/
+          /* CUA has Ctrl-Up/Ctrl-Down as paragraph up down */
+		case GDK_Up:
+			move_cursor_ver (html, -1);
+			break;
+		case GDK_Down:
+			move_cursor_ver (html, +1);
+			break;
+		case GDK_Left:
+			move_cursor_hor (html, -1);
+			break;
+		case GDK_Right:
+			move_cursor_hor (html, +1);
+			break;
+#if 0
+		case GDK_BackSpace:
+			if (event->state & GDK_CONTROL_MASK)
+				gtk_text_delete_backward_word (text);
+			else
+				gtk_text_delete_backward_character (text);
+			break;
+		case GDK_Clear:
+			gtk_text_delete_line (text);
+			break;
+		case GDK_Insert:
+			if (event->state & GDK_SHIFT_MASK)
+			{
+				extend_selection = FALSE;
+				gtk_editable_paste_clipboard (editable);
+			}
+			else if (event->state & GDK_CONTROL_MASK)
+			{
+				gtk_editable_copy_clipboard (editable);
+			}
+			else
+			{
+				/* gtk_toggle_insert(text) -- IMPLEMENT */
+			}
+			break;
+		case GDK_Delete:
+			if (event->state & GDK_CONTROL_MASK)
+				gtk_text_delete_forward_word (text);
+			else if (event->state & GDK_SHIFT_MASK)
+			{
+				extend_selection = FALSE;
+				gtk_editable_cut_clipboard (editable);
+			}
+			else
+				gtk_text_delete_forward_character (text);
+			break;
+		case GDK_Tab:
+			position = text->point.index;
+			gtk_editable_insert_text (editable, "\t", 1, &position);
+			break;
+		case GDK_Return:
+			if (event->state & GDK_CONTROL_MASK)
+				gtk_signal_emit_by_name (GTK_OBJECT (text), "activate");
+			else
+			{
+				position = text->point.index;
+				gtk_editable_insert_text (editable, "\n", 1, &position);
+			}
+			break;
+		case GDK_Escape:
+			/* Don't insert literally */
+			return_val = FALSE;
+			break;
+#endif
+		default:
+			return_val = FALSE;
+
+#if 0
+			if (event->state & GDK_CONTROL_MASK) {
+				if ((key >= 'A') && (key <= 'Z'))
+					key -= 'A' - 'a';
+
+				if ((key >= 'a') && (key <= 'z') && control_keys[(int) (key - 'a')])
+				{
+					(* control_keys[(int) (key - 'a')]) (editable, event->time);
+					return_val = TRUE;
+				}
+
+				break;
+			}
+			else if (event->state & GDK_MOD1_MASK)
+			{
+				if ((key >= 'A') && (key <= 'Z'))
+					key -= 'A' - 'a';
+
+				if ((key >= 'a') && (key <= 'z') && alt_keys[(int) (key - 'a')])
+				{
+					(* alt_keys[(int) (key - 'a')]) (editable, event->time);
+					return_val = TRUE;
+				}
+				break;
+			}
+#endif
+			/*
+			 if (event->length > 0) {
+			 html->cursor_pos++;
+			 gtk_editable_insert_text (editable, event->string, event->length, &position);
+
+			 return_val = TRUE;
+			 }
+			 else
+			 return_val = FALSE;
+			*/
+		}
+
+	}
+
+	return return_val;
+}
+ 
+void
+gtk_html_freeze (GtkHtml *html)
+{
+	g_return_if_fail (html != NULL);
+	g_return_if_fail (GTK_IS_HTML (html));
+
+	html->frozen++;
+}
+
+void
+gtk_html_thaw (GtkHtml *html)
+{
+	GdkRectangle area;
+	
+	g_return_if_fail (html != NULL);
+	g_return_if_fail (GTK_IS_HTML (html));
+
+	html->frozen--;
+
+	if (html->frozen < 0)
+                html->frozen = 0;
+
+	if (html->frozen == 0) {
+                if (html->html_area) {
+                        gint width, height;
+			area.x = 0;
+			area.y = 0;
+
+			gdk_window_get_size(html->html_area, &width, &height);
+
+                        area.width = width;
+                        area.height = height;
+                        
+			expose_html(html, &area, TRUE);
+		}
+	}
+}