--- /dev/null
+/* X-Chat
+ * Copyright (C) 1998 Peter Zelezny.
+ *
+ * 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
+ *
+ * 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
+ * =========================================================================
+ *
+ * xtext, the text widget used by X-Chat.
+ *
+ * By Peter Zelezny <zed@linux.com>.
+ * Some functions used from Zvt and Eterm (transparency stuff).
+ *
+ */
+
+#define USE_XLIB /* turn this ON for non-xchat use. */
+#undef XCHAT /* using xchat */
+#define REFRESH_TIMEOUT 20
+#define WORDWRAP_LIMIT 24
+#define TINT_VALUE 195 /* 195/255 of the brightness. */
+#define MOTION_MONITOR 1 /* URL hilights. */
+#define MARGIN 2 /* dont touch. */
+
+#include <config.h> /* can define USE_XLIB here */
+#include <string.h>
+#include <ctype.h>
+#include <stdlib.h>
+#include <stdio.h>
+#include <time.h>
+#include <gtk/gtkmain.h>
+#include <gtk/gtksignal.h>
+#include <gtk/gtkselection.h>
+
+#ifdef USE_XLIB
+#include <gdk/gdkx.h>
+#include <X11/Xlib.h>
+#include <X11/Xatom.h>
+#endif
+
+#include "xtext.h"
+
+#ifdef USE_GDK_PIXBUF
+#include <gdk-pixbuf/gdk-pixbuf.h>
+#endif
+
+#undef GTK_WIDGET
+#define GTK_WIDGET(n) ((GtkWidget*)n)
+#undef GTK_OBJECT
+#define GTK_OBJECT(n) ((GtkObject*)n)
+#undef GTK_OBJECT_CLASS
+#define GTK_OBJECT_CLASS(n) ((GtkObjectClass*)n)
+
+static GtkWidgetClass *parent_class = NULL;
+
+enum
+{
+ WORD_CLICK,
+ LAST_SIGNAL
+};
+static guint xtext_signals[LAST_SIGNAL] = { 0 };
+
+#ifdef XCHAT
+char *nocasestrstr (char *text, char *tofind); /* util.c */
+#endif
+static void gtk_xtext_render_page (GtkXText * xtext);
+static void gtk_xtext_calc_lines (GtkXText * xtext, int);
+#ifdef USE_XLIB
+static void gtk_xtext_load_trans (GtkXText * xtext);
+static void gtk_xtext_free_trans (GtkXText * xtext);
+#endif
+static textentry *gtk_xtext_nth (GtkXText * xtext, textentry * start_ent,
+ int line, int width, int *subline);
+static gint gtk_xtext_selection_kill (GtkWidget * widget,
+ GdkEventSelection * event);
+static void gtk_xtext_selection_get (GtkWidget * widget,
+ GtkSelectionData * selection_data_ptr,
+ guint info, guint time);
+static int gtk_xtext_text_width (GtkXText * xtext, unsigned char *text,
+ int len);
+static void gtk_xtext_adjustment_changed (GtkAdjustment * adj,
+ GtkXText * xtext);
+static void gtk_xtext_draw_sep (GtkXText * xtext, int height);
+static void gtk_xtext_render_ents (GtkXText * xtext, textentry *, textentry *,
+ int);
+static void gtk_xtext_recalc_widths (GtkXText * xtext, int);
+static void gtk_xtext_fix_indent (GtkXText * xtext);
+
+/* some utility functions first */
+
+#ifndef XCHAT /* xchat has this in util.c */
+
+static char *
+nocasestrstr (char *s, char *wanted)
+{
+ register const size_t len = strlen (wanted);
+
+ if (len == 0)
+ return (char *)s;
+ while (toupper(*s) != toupper(*wanted) || strncasecmp (s, wanted, len))
+ if (*s++ == '\0')
+ return (char *)NULL;
+ return (char *)s;
+}
+
+#endif
+
+static int
+is_del (char c)
+{
+ switch (c)
+ {
+ case ' ':
+ case 0:
+ case '\n':
+ /*case '[':
+ case ']': */
+ case ')':
+ case '(':
+ case '>':
+ case '<':
+ return 1;
+ }
+ return 0;
+}
+
+static void
+xtext_set_fg (GdkGC *gc, gulong pixel)
+{
+ GdkColor col;
+
+ col.pixel = pixel;
+ gdk_gc_set_foreground (gc, &col);
+}
+
+static void
+xtext_set_bg (GdkGC *gc, gulong pixel)
+{
+ GdkColor col;
+
+ col.pixel = pixel;
+ gdk_gc_set_background (gc, &col);
+}
+
+static void
+gtk_xtext_init (GtkXText * xtext)
+{
+ xtext->old_value = -1;
+ xtext->pixmap = NULL;
+ xtext->text_first = NULL;
+ xtext->text_last = NULL;
+ xtext->io_tag = -1;
+ xtext->add_io_tag = -1;
+ xtext->scroll_tag = -1;
+/* xtext->frozen = 0;*/
+ xtext->num_lines = 0;
+ xtext->max_lines = 0;
+ xtext->col_back = 19;
+ xtext->col_fore = 18;
+ xtext->nc = 0;
+ xtext->scrollbar_down = TRUE;
+ xtext->bold = FALSE;
+ xtext->underline = FALSE;
+ xtext->reverse = FALSE;
+ xtext->time_stamp = FALSE;
+ xtext->font = NULL;
+ xtext->error_function = NULL;
+ xtext->urlcheck_function = NULL;
+ xtext->color_paste = FALSE;
+ xtext->skip_fills = FALSE;
+ xtext->skip_border_fills = FALSE;
+ xtext->do_underline_fills_only = FALSE;
+ xtext->tint_red = xtext->tint_green = xtext->tint_blue = TINT_VALUE;
+
+ xtext->adj = (GtkAdjustment *) gtk_adjustment_new (0, 0, 0, 1, 0, 0);
+ gtk_object_ref ((GtkObject *) xtext->adj);
+ gtk_object_sink ((GtkObject *) xtext->adj);
+
+ gtk_signal_connect (GTK_OBJECT (xtext->adj), "value_changed",
+ GTK_SIGNAL_FUNC (gtk_xtext_adjustment_changed), xtext);
+ gtk_signal_connect (GTK_OBJECT (xtext), "selection_clear_event",
+ GTK_SIGNAL_FUNC (gtk_xtext_selection_kill), xtext);
+ gtk_selection_add_target (GTK_WIDGET (xtext),
+ GDK_SELECTION_PRIMARY,
+ GDK_SELECTION_TYPE_STRING, 1);
+ gtk_signal_connect (GTK_OBJECT (xtext), "selection_get",
+ GTK_SIGNAL_FUNC (gtk_xtext_selection_get), xtext);
+}
+
+static void
+gtk_xtext_adjustment_set (GtkXText * xtext, int fire_signal)
+{
+ GtkAdjustment *adj = xtext->adj;
+
+ adj->lower = 0;
+ adj->upper = xtext->num_lines;
+
+ adj->page_size =
+ (GTK_WIDGET (xtext)->allocation.height -
+ xtext->font->descent) / xtext->fontsize;
+ adj->page_increment = adj->page_size;
+
+ if (adj->value > adj->upper - adj->page_size)
+ adj->value = adj->upper - adj->page_size;
+
+ if (fire_signal)
+ gtk_adjustment_changed (adj);
+}
+
+static gint
+gtk_xtext_adjustment_timeout (GtkXText * xtext)
+{
+ gtk_xtext_render_page (xtext);
+ xtext->io_tag = -1;
+ return 0;
+}
+
+static void
+gtk_xtext_adjustment_changed (GtkAdjustment * adj, GtkXText * xtext)
+{
+/* if (xtext->frozen)
+ return;*/
+
+ if ((int) xtext->old_value != (int) xtext->adj->value)
+ {
+ if (xtext->adj->value >= xtext->adj->upper - xtext->adj->page_size)
+ xtext->scrollbar_down = TRUE;
+ else
+ xtext->scrollbar_down = FALSE;
+
+ if (xtext->adj->value + 1 == xtext->old_value ||
+ xtext->adj->value - 1 == xtext->old_value) /* clicked an arrow? */
+ {
+ if (xtext->io_tag != -1)
+ {
+ gtk_timeout_remove (xtext->io_tag);
+ xtext->io_tag = -1;
+ }
+ gtk_xtext_render_page (xtext);
+ } else
+ {
+ if (xtext->io_tag == -1)
+ xtext->io_tag = gtk_timeout_add (REFRESH_TIMEOUT,
+ (GtkFunction)
+ gtk_xtext_adjustment_timeout,
+ xtext);
+ }
+ }
+ xtext->old_value = adj->value;
+}
+
+GtkWidget *
+gtk_xtext_new (int indent, int separator)
+{
+ GtkXText *xtext;
+
+ xtext = gtk_type_new (gtk_xtext_get_type ());
+ xtext->indent = indent;
+ xtext->separator = separator;
+ xtext->wordwrap = FALSE;
+ xtext->double_buffer = FALSE;
+
+ return GTK_WIDGET (xtext);
+}
+
+static void
+gtk_xtext_destroy (GtkObject * object)
+{
+ GtkXText *xtext = GTK_XTEXT (object);
+ textentry *ent, *next;
+
+ if (xtext->add_io_tag != -1)
+ {
+ gtk_timeout_remove (xtext->add_io_tag);
+ xtext->add_io_tag = -1;
+ }
+
+ if (xtext->scroll_tag != -1)
+ {
+ gtk_timeout_remove (xtext->scroll_tag);
+ xtext->scroll_tag = -1;
+ }
+
+ if (xtext->io_tag != -1)
+ {
+ gtk_timeout_remove (xtext->io_tag);
+ xtext->io_tag = -1;
+ }
+
+ if (xtext->pixmap)
+ {
+#ifdef USE_XLIB
+ if (xtext->transparent)
+ gtk_xtext_free_trans (xtext);
+ else
+#endif
+ gdk_pixmap_unref (xtext->pixmap);
+ xtext->pixmap = NULL;
+ }
+
+ if (xtext->font)
+ {
+ gdk_font_unref (xtext->font);
+ xtext->font = NULL;
+ }
+
+ if (xtext->adj)
+ {
+ gtk_signal_disconnect_by_data (GTK_OBJECT (xtext->adj), xtext);
+ gtk_object_unref (GTK_OBJECT (xtext->adj));
+ xtext->adj = NULL;
+ }
+
+ if (xtext->bgc)
+ {
+ gdk_gc_destroy (xtext->bgc);
+ xtext->bgc = NULL;
+ }
+
+ if (xtext->fgc)
+ {
+ gdk_gc_destroy (xtext->fgc);
+ xtext->fgc = NULL;
+ }
+
+ if (xtext->light_gc)
+ {
+ gdk_gc_destroy (xtext->light_gc);
+ xtext->light_gc = NULL;
+ }
+
+ if (xtext->dark_gc)
+ {
+ gdk_gc_destroy (xtext->dark_gc);
+ xtext->dark_gc = NULL;
+ }
+
+ if (xtext->hand_cursor)
+ {
+ gdk_cursor_destroy (xtext->hand_cursor);
+ xtext->hand_cursor = NULL;
+ }
+
+ ent = xtext->text_first;
+ while (ent)
+ {
+ next = ent->next;
+ free (ent);
+ ent = next;
+ }
+ xtext->text_first = NULL;
+
+ if (GTK_OBJECT_CLASS (parent_class)->destroy)
+ (*GTK_OBJECT_CLASS (parent_class)->destroy) (object);
+}
+
+static void
+gtk_xtext_realize (GtkWidget * widget)
+{
+ GtkXText *xtext;
+ GdkWindowAttr attributes;
+ GdkGCValues val;
+ GdkColor col;
+ GdkColormap *cmap;
+
+ GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
+ xtext = GTK_XTEXT (widget);
+
+ 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.window_type = GDK_WINDOW_CHILD;
+ attributes.event_mask = gtk_widget_get_events (widget) |
+ GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK
+#ifdef MOTION_MONITOR
+ | GDK_POINTER_MOTION_MASK | GDK_LEAVE_NOTIFY_MASK;
+#else
+ | GDK_POINTER_MOTION_MASK;
+#endif
+
+ cmap = gtk_widget_get_colormap (widget);
+ attributes.colormap = cmap;
+ attributes.visual = gtk_widget_get_visual (widget);
+
+ widget->window = gdk_window_new (widget->parent->window, &attributes,
+ GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL |
+ GDK_WA_COLORMAP);
+
+ gdk_window_set_user_data (widget->window, widget);
+
+ xtext->depth = gdk_window_get_visual (widget->window)->depth;
+
+ val.subwindow_mode = GDK_INCLUDE_INFERIORS;
+ val.graphics_exposures = 0;
+
+ xtext->bgc = gdk_gc_new_with_values (widget->window, &val,
+ GDK_GC_EXPOSURES | GDK_GC_SUBWINDOW);
+ xtext->fgc = gdk_gc_new_with_values (widget->window, &val,
+ GDK_GC_EXPOSURES | GDK_GC_SUBWINDOW);
+ xtext->light_gc = gdk_gc_new_with_values (widget->window, &val,
+ GDK_GC_EXPOSURES | GDK_GC_SUBWINDOW);
+ xtext->dark_gc = gdk_gc_new_with_values (widget->window, &val,
+ GDK_GC_EXPOSURES | GDK_GC_SUBWINDOW);
+
+ /* for the separator bar (light) */
+ col.red = 0xffff; col.green = 0xffff; col.blue = 0xffff;
+ /* is setting the pixel necessary (or even correct) ?? */
+ col.pixel = (gulong)((col.red & 0xff00) * 256 +
+ (col.green & 0xff00) +
+ (col.blue & 0xff00) / 256);
+ gdk_color_alloc (cmap, &col);
+ gdk_gc_set_foreground (xtext->light_gc, &col);
+
+ /* for the separator bar (dark) */
+ col.red = 0x8e38; col.green = 0x8e38; col.blue = 0x9f38;
+ col.pixel = (gulong)((col.red & 0xff00) * 256 +
+ (col.green & 0xff00) +
+ (col.blue & 0xff00) / 256);
+ gdk_color_alloc (cmap, &col);
+ gdk_gc_set_foreground (xtext->dark_gc, &col);
+
+ if (xtext->fonttype != FONT_SET && xtext->font != NULL)
+ gdk_gc_set_font (xtext->fgc, xtext->font);
+
+ xtext_set_fg (xtext->fgc, xtext->palette[18]);
+ xtext_set_bg (xtext->fgc, xtext->palette[19]);
+ xtext_set_fg (xtext->bgc, xtext->palette[19]);
+
+#ifdef USE_XLIB
+ if (xtext->transparent)
+ {
+ gtk_xtext_load_trans (xtext);
+ } else if (xtext->pixmap)
+ {
+ gdk_gc_set_tile (xtext->bgc, xtext->pixmap);
+ gdk_gc_set_ts_origin (xtext->bgc, 0, 0);
+ gdk_gc_set_fill (xtext->bgc, GDK_TILED);
+ }
+#else
+ if (xtext->pixmap)
+ {
+ gdk_gc_set_tile (xtext->bgc, xtext->pixmap);
+ gdk_gc_set_ts_origin (xtext->bgc, 0, 0);
+ gdk_gc_set_fill (xtext->bgc, GDK_TILED);
+ }
+#endif
+
+ xtext->hand_cursor = gdk_cursor_new (GDK_HAND1);
+
+ gdk_window_set_back_pixmap (widget->window, NULL, FALSE);
+
+ /* if not doublebuffer, draw directly to window */
+ if (!xtext->double_buffer)
+ xtext->draw_buf = widget->window;
+
+ if (xtext->auto_indent)
+ xtext->indent = 1;
+}
+
+static void
+gtk_xtext_size_request (GtkWidget * widget, GtkRequisition * requisition)
+{
+ requisition->width = GTK_XTEXT (widget)->fontwidth['Z'] * 20;
+ requisition->height = (GTK_XTEXT (widget)->fontsize * 10) + 3;
+}
+
+static void
+gtk_xtext_size_allocate (GtkWidget * widget, GtkAllocation * allocation)
+{
+ GtkXText *xtext = GTK_XTEXT (widget);
+
+ if (allocation->width == widget->allocation.width &&
+ allocation->height == widget->allocation.height &&
+ allocation->x == widget->allocation.x &&
+ allocation->y == widget->allocation.y)
+ return;
+
+ widget->allocation = *allocation;
+ if (GTK_WIDGET_REALIZED (widget))
+ {
+ gdk_window_move_resize (widget->window,
+ allocation->x, allocation->y,
+ allocation->width, allocation->height);
+ gtk_xtext_calc_lines (xtext, FALSE);
+ }
+}
+
+static void
+gtk_xtext_draw (GtkWidget * widget, GdkRectangle * area)
+{
+ int x, y;
+ GtkXText *xtext = GTK_XTEXT (widget);
+
+#ifdef USE_XLIB
+ if (xtext->transparent)
+ {
+ gdk_window_get_origin (widget->window, &x, &y);
+ /* update transparency only if it moved */
+ if (xtext->last_win_x != x || xtext->last_win_y != y)
+ {
+ xtext->last_win_x = x;
+ xtext->last_win_y = y;
+ gtk_xtext_free_trans (xtext);
+ gtk_xtext_load_trans (xtext);
+ }
+ }
+#endif
+
+ if (xtext->scrollbar_down)
+ gtk_adjustment_set_value (xtext->adj,
+ xtext->adj->upper - xtext->adj->page_size);
+ gtk_xtext_render_page (xtext);
+}
+
+static int
+gtk_xtext_selection_clear (GtkXText * xtext)
+{
+ textentry *ent;
+ int ret = 0;
+
+ ent = xtext->last_ent_start;
+ while (ent)
+ {
+ if (ent->mark_start != -1)
+ ret = 1;
+ ent->mark_start = -1;
+ ent->mark_end = -1;
+ if (ent == xtext->last_ent_end)
+ break;
+ ent = ent->next;
+ }
+
+ return ret;
+}
+
+static int
+find_x_8bit (GtkXText *xtext, textentry *ent, char *text, int x, int indent)
+{
+ int xx = indent;
+ int i = 0;
+ int col = FALSE;
+ int nc = 0;
+ char *orig = text;
+ int a;
+
+ while (*text)
+ {
+ if ((col && isdigit (*text) && nc < 2) ||
+ (col && *text == ',' && nc < 3))
+ {
+ nc++;
+ if (*text == ',')
+ nc = 0;
+ } else
+ {
+ col = FALSE;
+ switch (*text)
+ {
+ case ATTR_COLOR:
+ col = TRUE;
+ nc = 0;
+ break;
+ case ATTR_BEEP:
+ case ATTR_RESET:
+ case ATTR_REVERSE:
+ case ATTR_BOLD:
+ case ATTR_UNDERLINE:
+ break;
+ default:
+ a = *((unsigned char *)text);
+ xx += xtext->fontwidth[a];
+ if (xx >= x)
+ return i + (orig - ent->str);
+ }
+ }
+ text++;
+ i++;
+ if (text - orig >= ent->str_len)
+ return ent->str_len;
+ }
+
+ return ent->str_len;
+}
+
+static int
+find_x_general (GtkXText * xtext, textentry * ent, char *str, int x, int indent)
+{
+ int str_width;
+ int len = 1;
+
+ while (1)
+ {
+ str_width = gtk_xtext_text_width (xtext, str, len);
+ if (str_width + indent >= x)
+ return (str + len) - ent->str;
+ len++;
+ if (len + (str - ent->str) > ent->str_len)
+ return ent->str_len;
+ if (str_width + indent + 40 < x)
+ len += 2;
+ }
+}
+
+static int
+find_x (GtkXText * xtext, textentry * ent, char *str, int x, int indent)
+{
+ if (xtext->fonttype == FONT_1BYTE)
+ return find_x_8bit (xtext, ent, str, x, indent);
+
+ return find_x_general (xtext, ent, str, x, indent);
+}
+
+static int
+gtk_xtext_find_x (GtkXText * xtext, int x, textentry * ent, int offset,
+ int line, int win_width, int *out_of_bounds)
+{
+ int indent;
+ char *str;
+
+ if (offset < 1)
+ indent = ent->indent;
+ else
+ indent = xtext->indent;
+
+ if (line > xtext->adj->page_size || line < 0)
+ return 0;
+
+ if (xtext->grid_offset[line] > ent->str_len)
+ return 0;
+
+ if (xtext->grid_offset[line] < 0)
+ return 0;
+
+ str = ent->str + xtext->grid_offset[line];
+
+ if (x < indent)
+ {
+ *out_of_bounds = 1;
+ return (str - ent->str);
+ }
+
+ *out_of_bounds = 0;
+
+ return find_x (xtext, ent, str, x, indent);
+}
+
+static textentry *
+gtk_xtext_find_char (GtkXText * xtext, int x, int y, int *off,
+ int *out_of_bounds)
+{
+ textentry *ent;
+ int line;
+ int subline;
+ int win_width;
+
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, &win_width, 0);
+ win_width -= MARGIN;
+
+ line = (y - xtext->font->descent) / xtext->fontsize;
+
+ subline = xtext->pagetop_subline;
+ ent = gtk_xtext_nth (xtext, xtext->pagetop_ent, line, win_width, &subline);
+ if (!ent)
+ return 0;
+
+ if (off)
+ *off = gtk_xtext_find_x (xtext, x, ent, subline, line, win_width,
+ out_of_bounds);
+
+ return ent;
+}
+
+static gint
+gtk_xtext_expose (GtkWidget * widget, GdkEventExpose * event)
+{
+ GtkXText *xtext = GTK_XTEXT (widget);
+ textentry *ent_start, *ent_end;
+
+ if (xtext->double_buffer)
+ {
+ gtk_xtext_render_page (xtext);
+ return FALSE;
+ }
+
+ gdk_draw_rectangle (xtext->draw_buf, xtext->bgc, 1,
+ event->area.x, event->area.y,
+ event->area.width, event->area.height);
+
+ ent_start = gtk_xtext_find_char (xtext, event->area.x, event->area.y,
+ NULL, NULL);
+ ent_end = gtk_xtext_find_char (xtext, event->area.x + event->area.width,
+ event->area.y + event->area.height, NULL, NULL);
+
+ xtext->skip_fills = TRUE;
+ xtext->skip_border_fills = TRUE;
+
+ gtk_xtext_render_ents (xtext, ent_start, ent_end, TRUE);
+
+ xtext->skip_fills = FALSE;
+ xtext->skip_border_fills = FALSE;
+
+ return FALSE;
+}
+
+static void
+gtk_xtext_selection_draw (GtkXText * xtext, GdkEventMotion * event)
+{
+ textentry *ent;
+ textentry *ent_end;
+ textentry *ent_start;
+ int offset_start;
+ int offset_end;
+ int low_x;
+ int low_y;
+ int high_x;
+ int high_y;
+ int tmp;
+
+ if (xtext->select_start_y > xtext->select_end_y)
+ {
+ low_x = xtext->select_end_x;
+ low_y = xtext->select_end_y;
+ high_x = xtext->select_start_x;
+ high_y = xtext->select_start_y;
+ } else
+ {
+ low_x = xtext->select_start_x;
+ low_y = xtext->select_start_y;
+ high_x = xtext->select_end_x;
+ high_y = xtext->select_end_y;
+ }
+
+ ent_start = gtk_xtext_find_char (xtext, low_x, low_y, &offset_start, &tmp);
+ ent_end = gtk_xtext_find_char (xtext, high_x, high_y, &offset_end, &tmp);
+
+ if (ent_start && !ent_end)
+ {
+ ent_end = xtext->text_last;
+ offset_end = ent_end->str_len;
+ }
+
+ if (!ent_start || !ent_end)
+ {
+ if (xtext->adj->value != xtext->old_value)
+ gtk_xtext_render_page (xtext);
+ return;
+ }
+
+ gtk_xtext_selection_clear (xtext);
+
+ /* marking less than a complete line? */
+ if (ent_start == ent_end)
+ {
+ ent_start->mark_start = MIN (offset_start, offset_end);
+ ent_start->mark_end = MAX (offset_end, offset_start);
+ if (offset_start == offset_end)
+ ent_start->mark_end++;
+ } else
+ {
+ ent_start->mark_start = offset_start;
+ ent_start->mark_end = ent_start->str_len;
+
+ if (offset_end != 0)
+ {
+ ent_end->mark_start = 0;
+ ent_end->mark_end = offset_end;
+ }
+ }
+
+ if (ent_start != ent_end)
+ {
+ ent = ent_start->next;
+ while (ent && ent != ent_end)
+ {
+ ent->mark_start = 0;
+ ent->mark_end = ent->str_len;
+ ent = ent->next;
+ }
+ }
+
+ /* has the selection changed? Dont render unless necessary */
+ if (xtext->last_ent_start == ent_start &&
+ xtext->last_ent_end == ent_end &&
+ xtext->last_offset_start == offset_start &&
+ xtext->last_offset_end == offset_end)
+ return;
+
+ gtk_selection_owner_set (GTK_WIDGET (xtext), GDK_SELECTION_PRIMARY,
+ event->time);
+
+ if (xtext->double_buffer)
+ {
+ if (xtext->io_tag == -1)
+ xtext->io_tag = gtk_timeout_add (REFRESH_TIMEOUT,
+ (GtkFunction)
+ gtk_xtext_adjustment_timeout,
+ xtext);
+ } else
+ {
+ ent = xtext->last_ent_end;
+ if (ent)
+ if (ent->next == ent_end)
+ ent = ent_end;
+ xtext->skip_border_fills = TRUE;
+ gtk_xtext_render_ents (xtext, xtext->last_ent_start, ent, TRUE);
+ xtext->skip_border_fills = FALSE;
+ xtext->old_ent_start = xtext->last_ent_start;
+ xtext->old_ent_end = xtext->last_ent_end;
+ }
+
+ xtext->last_ent_start = ent_start;
+ xtext->last_ent_end = ent_end;
+ xtext->last_offset_start = offset_start;
+ xtext->last_offset_end = offset_end;
+}
+
+static gint
+gtk_xtext_scrolldown_timeout (GtkXText * xtext)
+{
+ int p_y, win_height;
+
+ gdk_window_get_pointer (GTK_WIDGET (xtext)->window, 0, &p_y, 0);
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, 0, &win_height);
+
+ if (p_y > win_height &&
+ xtext->adj->value < (xtext->adj->upper - xtext->adj->page_size))
+ {
+ xtext->adj->value++;
+ gtk_adjustment_changed (xtext->adj);
+ gtk_xtext_render_page (xtext);
+ return 1;
+ }
+
+ xtext->scroll_tag = -1;
+ return 0;
+}
+
+static gint
+gtk_xtext_scrollup_timeout (GtkXText * xtext)
+{
+ int p_y;
+
+ gdk_window_get_pointer (GTK_WIDGET (xtext)->window, 0, &p_y, 0);
+
+ if (p_y < 0 && xtext->adj->value > 0.0)
+ {
+ xtext->adj->value--;
+ gtk_adjustment_changed (xtext->adj);
+ gtk_xtext_render_page (xtext);
+ return 1;
+ }
+
+ xtext->scroll_tag = -1;
+ return 0;
+}
+
+static void
+gtk_xtext_selection_update (GtkXText * xtext, GdkEventMotion * event, int p_y)
+{
+ int win_height;
+ int moved;
+
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, 0, &win_height);
+
+ /* selecting past top of window, scroll up! */
+ if (p_y < 0 && xtext->adj->value >= 0)
+ {
+ if (xtext->scroll_tag == -1)
+ xtext->scroll_tag = gtk_timeout_add (100,
+ (GtkFunction)
+ gtk_xtext_scrollup_timeout,
+ xtext);
+ return;
+ }
+
+ /* selecting past bottom of window, scroll down! */
+ if (p_y > win_height &&
+ xtext->adj->value < (xtext->adj->upper - xtext->adj->page_size))
+ {
+ if (xtext->scroll_tag == -1)
+ xtext->scroll_tag = gtk_timeout_add (100,
+ (GtkFunction)
+ gtk_xtext_scrolldown_timeout,
+ xtext);
+ return;
+ }
+
+ moved = xtext->adj->value - xtext->select_start_adj;
+ xtext->select_start_y -= (moved * xtext->fontsize);
+ xtext->select_start_adj = xtext->adj->value;
+ gtk_xtext_selection_draw (xtext, event);
+}
+
+static char *
+gtk_xtext_get_word (GtkXText * xtext, int x, int y, textentry ** ret_ent,
+ int *ret_off, int *ret_len)
+{
+ textentry *ent;
+ int offset;
+ char *str;
+ char *word;
+ int len;
+ int out_of_bounds;
+
+ ent = gtk_xtext_find_char (xtext, x, y, &offset, &out_of_bounds);
+ if (!ent)
+ return 0;
+
+ if (out_of_bounds)
+ return 0;
+
+ if (offset == ent->str_len)
+ return 0;
+
+ if (offset < 1)
+ return 0;
+
+ offset--;
+
+ str = ent->str + offset;
+
+ while (!is_del (*str) && str != ent->str)
+ str--;
+ word = str + 1;
+
+ len = 0;
+ str = word;
+ while (!is_del (*str) && len != ent->str_len)
+ {
+ str++;
+ len++;
+ }
+
+ if (ret_ent)
+ *ret_ent = ent;
+ if (ret_off)
+ *ret_off = word - ent->str;
+ if (ret_len)
+ *ret_len = str - word;
+
+ word = gtk_xtext_strip_color (word, len, NULL, NULL);
+
+ return word;
+}
+
+static gint
+gtk_xtext_leave_notify (GtkWidget * widget, GdkEventCrossing * event)
+{
+#ifdef MOTION_MONITOR
+ GtkXText *xtext = GTK_XTEXT (widget);
+
+ if (xtext->cursor_hand)
+ {
+ xtext->hilight_start = -1;
+ xtext->hilight_end = -1;
+ xtext->cursor_hand = FALSE;
+ gdk_window_set_cursor (widget->window, 0);
+ xtext->skip_border_fills = TRUE;
+ xtext->do_underline_fills_only = TRUE;
+ gtk_xtext_render_ents (xtext, xtext->hilight_ent, NULL, FALSE);
+ xtext->skip_border_fills = FALSE;
+ xtext->do_underline_fills_only = FALSE;
+ xtext->hilight_ent = NULL;
+ }
+#endif
+ return FALSE;
+}
+
+static gint
+gtk_xtext_motion_notify (GtkWidget * widget, GdkEventMotion * event)
+{
+ GtkXText *xtext = GTK_XTEXT (widget);
+ int tmp, x, y, offset, len;
+ char *word;
+ textentry *word_ent, *old_ent;
+
+ gdk_window_get_pointer (widget->window, &x, &y, 0);
+
+ if (xtext->moving_separator)
+ {
+ if (x < (3 * widget->allocation.width) / 5 && x > 15)
+ {
+ tmp = xtext->indent;
+ xtext->indent = x;
+ gtk_xtext_fix_indent (xtext);
+ if (tmp != xtext->indent)
+ {
+ gtk_xtext_recalc_widths (xtext, FALSE);
+ if (xtext->scrollbar_down)
+ gtk_adjustment_set_value (xtext->adj, xtext->adj->upper -
+ xtext->adj->page_size);
+ if (xtext->io_tag == -1)
+ xtext->io_tag = gtk_timeout_add (REFRESH_TIMEOUT,
+ (GtkFunction)
+ gtk_xtext_adjustment_timeout,
+ xtext);
+ }
+ }
+ return FALSE;
+ }
+
+ if (xtext->button_down)
+ {
+ gtk_grab_add (widget);
+ /*gdk_pointer_grab (widget->window, TRUE,
+ GDK_BUTTON_RELEASE_MASK |
+ GDK_BUTTON_MOTION_MASK, NULL, NULL, 0);*/
+ xtext->select_end_x = x;
+ xtext->select_end_y = y;
+ gtk_xtext_selection_update (xtext, event, y);
+ return FALSE;
+ }
+#ifdef MOTION_MONITOR
+
+ if (xtext->urlcheck_function == NULL)
+ return FALSE;
+
+ word = gtk_xtext_get_word (xtext, x, y, &word_ent, &offset, &len);
+ if (word)
+ {
+ if (xtext->urlcheck_function (xtext, word) > 0)
+ {
+ free (word);
+ if (!xtext->cursor_hand ||
+ xtext->hilight_ent != word_ent ||
+ xtext->hilight_start != offset ||
+ xtext->hilight_end != offset + len)
+ {
+ if (!xtext->cursor_hand)
+ {
+ gdk_window_set_cursor (GTK_WIDGET (xtext)->window,
+ xtext->hand_cursor);
+ xtext->cursor_hand = TRUE;
+ }
+ old_ent = xtext->hilight_ent;
+ xtext->hilight_ent = word_ent;
+ xtext->hilight_start = offset;
+ xtext->hilight_end = offset + len;
+ xtext->skip_border_fills = TRUE;
+ xtext->do_underline_fills_only = TRUE;
+ gtk_xtext_render_ents (xtext, old_ent, word_ent, FALSE);
+ xtext->skip_border_fills = FALSE;
+ xtext->do_underline_fills_only = FALSE;
+ }
+ return FALSE;
+ }
+ free (word);
+ }
+
+ gtk_xtext_leave_notify (widget, NULL);
+
+#endif
+
+ return FALSE;
+}
+
+static gint
+gtk_xtext_button_release (GtkWidget * widget, GdkEventButton * event)
+{
+ GtkXText *xtext = GTK_XTEXT (widget);
+ char *word;
+
+ if (xtext->moving_separator)
+ {
+ xtext->moving_separator = FALSE;
+ if (event->x < (4 * widget->allocation.width) / 5 && event->x > 15)
+ {
+ xtext->indent = event->x;
+ }
+ gtk_xtext_fix_indent (xtext);
+ gtk_xtext_recalc_widths (xtext, FALSE);
+ gtk_xtext_adjustment_set (xtext, TRUE);
+ gtk_xtext_render_page (xtext);
+ return FALSE;
+ }
+
+ if (xtext->word_or_line_select)
+ {
+ xtext->word_or_line_select = FALSE;
+ xtext->button_down = FALSE;
+ return FALSE;
+ }
+
+ if (event->button == 1)
+ {
+ xtext->button_down = FALSE;
+
+ gtk_grab_remove (widget);
+ /*gdk_pointer_ungrab (0);*/
+
+ if (xtext->select_start_x == event->x &&
+ xtext->select_start_y == event->y)
+ {
+ if (gtk_xtext_selection_clear (xtext))
+ gtk_xtext_render_page (xtext);
+ } else
+ {
+ word = gtk_xtext_get_word (xtext, event->x, event->y, 0, 0, 0);
+ if (word)
+ {
+ gtk_signal_emit (GTK_OBJECT (xtext), xtext_signals[WORD_CLICK],
+ word, event);
+ free (word);
+ return FALSE;
+ }
+ }
+ }
+
+ return FALSE;
+}
+
+static gint
+gtk_xtext_button_press (GtkWidget * widget, GdkEventButton * event)
+{
+ GtkXText *xtext = GTK_XTEXT (widget);
+ textentry *ent;
+ char *word;
+ int line_x, x, y, offset, len;
+ gfloat new_value;
+
+ gdk_window_get_pointer (widget->window, &x, &y, 0);
+
+ if (event->button == 3) /* right click */
+ {
+ word = gtk_xtext_get_word (xtext, x, y, 0, 0, 0);
+ if (word)
+ {
+ gtk_signal_emit (GTK_OBJECT (xtext), xtext_signals[WORD_CLICK],
+ word, event);
+ free (word);
+ } else
+ gtk_signal_emit (GTK_OBJECT (xtext), xtext_signals[WORD_CLICK],
+ "", event);
+ return FALSE;
+ }
+
+ if (event->button == 4) /* mouse wheel pageUp */
+ {
+ new_value = xtext->adj->value - xtext->adj->page_increment;
+ if (new_value < xtext->adj->lower)
+ new_value = xtext->adj->lower;
+ gtk_adjustment_set_value (xtext->adj, new_value);
+ return FALSE;
+ }
+
+ if (event->button == 5) /* mouse wheel pageDn */
+ {
+ new_value = xtext->adj->value + xtext->adj->page_increment;
+ if (new_value > (xtext->adj->upper - xtext->adj->page_size))
+ new_value = xtext->adj->upper - xtext->adj->page_size;
+ gtk_adjustment_set_value (xtext->adj, new_value);
+ return FALSE;
+ }
+
+ if (event->button == 2)
+ {
+ gtk_signal_emit (GTK_OBJECT (xtext), xtext_signals[WORD_CLICK], "", event);
+ return FALSE;
+ }
+
+ if (event->button != 1) /* we only want left button */
+ return FALSE;
+
+ if (event->type == GDK_2BUTTON_PRESS) /* WORD select */
+ {
+ word = gtk_xtext_get_word (xtext, x, y, &ent, &offset, &len);
+ if (word)
+ {
+ free (word);
+ if (len == 0)
+ return FALSE;
+ gtk_xtext_selection_clear (xtext);
+ ent->mark_start = offset;
+ ent->mark_end = offset + len;
+ xtext->last_ent_start = ent;
+ xtext->last_ent_end = ent;
+ gtk_xtext_render_page (xtext);
+ xtext->word_or_line_select = TRUE;
+ gtk_selection_owner_set (widget, GDK_SELECTION_PRIMARY, event->time);
+ }
+
+ return FALSE;
+ }
+
+ if (event->type == GDK_3BUTTON_PRESS) /* LINE select */
+ {
+ word = gtk_xtext_get_word (xtext, x, y, &ent, 0, 0);
+ if (word)
+ {
+ free (word);
+ gtk_xtext_selection_clear (xtext);
+ ent->mark_start = 0;
+ ent->mark_end = ent->str_len;
+ xtext->last_ent_start = ent;
+ xtext->last_ent_end = ent;
+ gtk_xtext_render_page (xtext);
+ xtext->word_or_line_select = TRUE;
+ gtk_selection_owner_set (widget, GDK_SELECTION_PRIMARY, event->time);
+ }
+
+ return FALSE;
+ }
+
+ /* check if it was a separator-bar click */
+ if (xtext->separator && xtext->indent)
+ {
+ line_x = xtext->indent - ((xtext->space_width + 1) / 2);
+ if (line_x == x || line_x == x + 1 || line_x == x - 1)
+ {
+ xtext->moving_separator = TRUE;
+ gtk_xtext_render_page (xtext);
+ return FALSE;
+ }
+ }
+
+ xtext->button_down = TRUE;
+
+ xtext->select_start_x = x;
+ xtext->select_start_y = y;
+
+ xtext->select_start_adj = xtext->adj->value;
+
+ return FALSE;
+}
+
+/* another program has claimed the selection */
+
+static gint
+gtk_xtext_selection_kill (GtkWidget * widget, GdkEventSelection * event)
+{
+ if (gtk_xtext_selection_clear (GTK_XTEXT (widget)))
+ gtk_xtext_render_page (GTK_XTEXT (widget));
+ return TRUE;
+}
+
+/* another program is asking for our selection */
+
+static void
+gtk_xtext_selection_get (GtkWidget * widget,
+ GtkSelectionData * selection_data_ptr,
+ guint info, guint time)
+{
+ GtkXText *xtext = GTK_XTEXT (widget);
+ textentry *ent;
+ char *txt;
+ char *pos;
+ char *stripped;
+ int len;
+ int first = TRUE;
+
+ /* first find out how much we need to malloc ... */
+ len = 0;
+ ent = xtext->text_first;
+ while (ent)
+ {
+ if (ent->mark_start != -1)
+ {
+ if (ent->mark_end - ent->mark_start > 0)
+ len += (ent->mark_end - ent->mark_start) + 1;
+ else
+ len++;
+ }
+ ent = ent->next;
+ }
+
+ /* now allocate mem and copy buffer */
+ pos = txt = malloc (len);
+ ent = xtext->text_first;
+ while (ent)
+ {
+ if (ent->mark_start != -1)
+ {
+ if (!first)
+ {
+ *pos = '\n';
+ pos++;
+ }
+ first = FALSE;
+ if (ent->mark_end - ent->mark_start > 0)
+ {
+ memcpy (pos, ent->str + ent->mark_start,
+ ent->mark_end - ent->mark_start);
+ pos += ent->mark_end - ent->mark_start;
+ }
+ }
+ ent = ent->next;
+ }
+ *pos = 0;
+
+ if (xtext->color_paste)
+ {
+ gtk_selection_data_set (selection_data_ptr, GDK_SELECTION_TYPE_STRING,
+ 8, txt, strlen (txt));
+ } else
+ {
+ stripped = gtk_xtext_strip_color (txt, strlen (txt), NULL, NULL);
+ gtk_selection_data_set (selection_data_ptr, GDK_SELECTION_TYPE_STRING,
+ 8, stripped, strlen (stripped));
+ free (stripped);
+ }
+
+ free (txt);
+}
+
+static void
+gtk_xtext_class_init (GtkXTextClass * class)
+{
+ GtkObjectClass *object_class;
+ GtkWidgetClass *widget_class;
+ GtkXTextClass *xtext_class;
+
+ object_class = (GtkObjectClass *) class;
+ widget_class = (GtkWidgetClass *) class;
+ xtext_class = (GtkXTextClass *) class;
+
+ parent_class = gtk_type_class (gtk_widget_get_type ());
+
+ xtext_signals[WORD_CLICK] =
+ gtk_signal_new (/*name*/"word_click",
+ /*GtkSignalRunType*/GTK_RUN_FIRST,
+ /*GtkType*/object_class->type,
+ /*funcoffset*/GTK_SIGNAL_OFFSET (GtkXTextClass, word_click),
+ /*GtkSignalMarshaller*/gtk_marshal_NONE__POINTER_POINTER,
+ /*returnval*/GTK_TYPE_NONE,
+ /*num args*/2, /*args*/GTK_TYPE_POINTER, GTK_TYPE_POINTER);
+ gtk_object_class_add_signals (object_class, xtext_signals, LAST_SIGNAL);
+
+ object_class->destroy = gtk_xtext_destroy;
+
+ widget_class->realize = gtk_xtext_realize;
+ widget_class->size_request = gtk_xtext_size_request;
+ widget_class->size_allocate = gtk_xtext_size_allocate;
+ widget_class->button_press_event = gtk_xtext_button_press;
+ widget_class->button_release_event = gtk_xtext_button_release;
+ widget_class->motion_notify_event = gtk_xtext_motion_notify;
+ widget_class->leave_notify_event = gtk_xtext_leave_notify;
+ widget_class->draw = gtk_xtext_draw;
+ widget_class->expose_event = gtk_xtext_expose;
+
+ xtext_class->word_click = NULL;
+}
+
+guint gtk_xtext_get_type ()
+{
+ static guint xtext_type = 0;
+
+ if (!xtext_type)
+ {
+ GtkTypeInfo xtext_info = {
+ "GtkXText",
+ sizeof (GtkXText),
+ sizeof (GtkXTextClass),
+ (GtkClassInitFunc) gtk_xtext_class_init,
+ (GtkObjectInitFunc) gtk_xtext_init,
+ (GtkArgSetFunc) NULL,
+ (GtkArgGetFunc) NULL,
+ };
+
+ xtext_type = gtk_type_unique (gtk_widget_get_type (), &xtext_info);
+ }
+
+ return xtext_type;
+}
+
+/*void
+gtk_xtext_thaw (GtkXText *xtext)
+{
+ if (xtext->frozen > 0)
+ xtext->frozen--;
+
+ if (xtext->frozen == 0)
+ gtk_xtext_render_page (xtext);
+}
+
+void
+gtk_xtext_freeze (GtkXText *xtext)
+{
+ xtext->frozen++;
+}*/
+
+/* strip MIRC colors and other attribs. */
+
+char *
+gtk_xtext_strip_color (unsigned char *text, int len, char *outbuf, int *newlen)
+{
+ int nc = 0;
+ int i = 0;
+ int col = FALSE;
+ char *new_str;
+
+ if (outbuf == NULL)
+ new_str = malloc (len + 2);
+ else
+ new_str = outbuf;
+
+ while (len > 0)
+ {
+ if ((col && isdigit (*text) && nc < 2) ||
+ (col && *text == ',' && nc < 3))
+ {
+ nc++;
+ if (*text == ',')
+ nc = 0;
+ } else
+ {
+ if (col)
+ col = FALSE;
+ switch (*text)
+ {
+ case ATTR_COLOR:
+ col = TRUE;
+ nc = 0;
+ break;
+ case ATTR_BEEP:
+ case ATTR_RESET:
+ case ATTR_REVERSE:
+ case ATTR_BOLD:
+ case ATTR_UNDERLINE:
+ break;
+ default:
+ new_str[i] = *text;
+ i++;
+ }
+ }
+ text++;
+ len--;
+ }
+
+ new_str[i] = 0;
+
+ if (newlen != NULL)
+ *newlen = i;
+
+ return new_str;
+}
+
+/* gives width of a 8bit string - with no mIRC codes in it */
+
+static int
+gtk_xtext_text_width_simple (GtkXText * xtext, unsigned char *str, int len)
+{
+ int width = 0;
+
+ if (xtext->fixed_width_font)
+ return (xtext->space_width * len);
+
+ while (len)
+ {
+ width += xtext->fontwidth[*str];
+ len--;
+ str++;
+ }
+
+ return width;
+}
+
+/* gives width of a string, excluding the mIRC codes */
+
+static int
+gtk_xtext_text_width (GtkXText * xtext, unsigned char *text, int len)
+{
+ unsigned char *tmp, *new_buf;
+ int width, new_len;
+
+ new_buf = gtk_xtext_strip_color (text, len, xtext->scratch_buffer, &new_len);
+
+ if (xtext->fonttype == FONT_1BYTE)
+ {
+ if (xtext->fixed_width_font)
+ {
+ width = xtext->space_width * new_len;
+ } else
+ {
+ width = 0;
+ tmp = new_buf;
+ while (*tmp)
+ {
+ width += xtext->fontwidth[*tmp];
+ tmp++;
+ }
+ }
+ } else
+ {
+ width = gdk_text_width (xtext->font, new_buf, new_len);
+ }
+
+ return width;
+}
+
+/* actually draw text to screen */
+
+static int
+gtk_xtext_render_flush (GtkXText * xtext, int x, int y, char *str, int len,
+ GdkGC *gc)
+{
+ int str_width;
+ int dofill;
+#ifdef USE_XLIB
+ XFontStruct *xfont;
+ GC xgc;
+ Drawable xdraw_buf;
+ Display *xdisplay;
+#endif
+
+ if (xtext->dont_render)
+ return 0;
+
+ if (xtext->fonttype == FONT_1BYTE)
+ str_width = gtk_xtext_text_width_simple (xtext, str, len);
+ else
+ str_width = gdk_text_width (xtext->font, str, len);
+
+ if (str_width < 1)
+ return 0;
+
+ if (!xtext->backcolor && xtext->pixmap)
+ {
+ dofill = FALSE;
+ /* draw the background pixmap behind the text - CAUSES FLICKER HERE !! */
+ if (!xtext->double_buffer && !xtext->skip_fills)
+ {
+ if (xtext->do_underline_fills_only)
+ {
+ gdk_draw_rectangle (GTK_WIDGET (xtext)->window, xtext->bgc, 1,
+ x, y + 1, str_width, 1);
+ if (xtext->underline) /* optimization */
+ goto dounder;
+ } else
+ {
+ gdk_draw_rectangle (GTK_WIDGET (xtext)->window, xtext->bgc, 1,
+ x, y - xtext->font->ascent, str_width,
+ xtext->fontsize);
+ }
+ }
+ } else
+ {
+ dofill = TRUE;
+ if (xtext->skip_fills && !xtext->backcolor)
+ dofill = FALSE;
+ }
+
+#ifdef USE_XLIB
+
+ xgc = GDK_GC_XGC (gc);
+ xdraw_buf = GDK_WINDOW_XWINDOW (xtext->draw_buf);
+ xdisplay = GDK_WINDOW_XDISPLAY (GTK_WIDGET (xtext)->window);
+ xfont = GDK_FONT_XFONT (xtext->font);
+
+ switch (xtext->fonttype)
+ {
+ case FONT_1BYTE:
+ if (dofill)
+ XDrawImageString (xdisplay, xdraw_buf, xgc, x, y, str, len);
+ else
+ XDrawString (xdisplay, xdraw_buf, xgc, x, y, str, len);
+ if (xtext->bold)
+ XDrawString (xdisplay, xdraw_buf, xgc, x + 1, y, str, len);
+ break;
+
+ case FONT_2BYTE:
+ len /= 2;
+ if (dofill)
+ XDrawImageString16 (xdisplay, xdraw_buf,
+ xgc, x, y, (XChar2b *) str, len);
+ else
+ XDrawString16 (xdisplay, xdraw_buf,
+ xgc, x, y, (XChar2b *) str, len);
+ if (xtext->bold)
+ XDrawString16 (xdisplay, xdraw_buf,
+ xgc, x + 1, y, (XChar2b *) str, len);
+ break;
+
+ case FONT_SET:
+ if (dofill)
+ XmbDrawImageString (xdisplay, xdraw_buf,
+ (XFontSet) xfont, xgc, x, y, str, len);
+ else
+ XmbDrawString (xdisplay, xdraw_buf,
+ (XFontSet) xfont, xgc, x, y, str, len);
+ if (xtext->bold)
+ XmbDrawString (xdisplay, xdraw_buf,
+ (XFontSet) xfont, xgc, x + 1, y, str, len);
+ }
+
+#else
+
+ /* don't have Xlib, gdk version --- */
+ if (dofill)
+ {
+ GdkGCValues val;
+ gdk_gc_get_values (gc, &val);
+ xtext_set_fg (gc, val.background.pixel);
+ gdk_draw_rectangle (xtext->draw_buf, gc, 1,
+ x, y - xtext->font->ascent, str_width,
+ xtext->fontsize);
+ xtext_set_fg (gc, val.foreground.pixel);
+ }
+ gdk_draw_text (xtext->draw_buf, xtext->font, gc, x, y, str, len);
+ if (xtext->bold)
+ gdk_draw_text (xtext->draw_buf, xtext->font, gc, x + 1, y, str, len);
+
+#endif
+
+ if (xtext->underline)
+dounder:
+ gdk_draw_line (xtext->draw_buf, gc, x, y+1, x+str_width-1, y+1);
+
+ return str_width;
+}
+
+static void
+gtk_xtext_reset (GtkXText * xtext, int mark, int attribs)
+{
+ if (attribs)
+ {
+ xtext->underline = FALSE;
+ xtext->bold = FALSE;
+ }
+ if (!mark)
+ {
+ xtext->backcolor = FALSE;
+ if (xtext->col_fore != 18)
+ xtext_set_fg (xtext->fgc, xtext->palette[18]);
+ if (xtext->col_back != 19)
+ xtext_set_bg (xtext->fgc, xtext->palette[19]);
+ }
+ xtext->col_fore = 18;
+ xtext->col_back = 19;
+}
+
+/* render a single line, which WONT wrap, and parse mIRC colors */
+
+static void
+gtk_xtext_render_str (GtkXText * xtext, int y, textentry * ent, char *str,
+ int len, int win_width, int indent, int line)
+{
+ GdkGC *gc;
+ int i = 0, x = indent, j = 0;
+ char *pstr = str;
+ int col_num, tmp;
+ int offset;
+ int mark = FALSE;
+ int hilight = FALSE;
+
+ offset = str - ent->str;
+
+ if (line < 255 && line >= 0)
+ xtext->grid_offset[line] = offset;
+
+ gc = xtext->fgc; /* our foreground GC */
+
+ if (ent->mark_start != -1 &&
+ ent->mark_start <= i + offset && ent->mark_end > i + offset)
+ {
+ xtext_set_bg (gc, xtext->palette[16]);
+ xtext_set_fg (gc, xtext->palette[17]);
+ xtext->backcolor = TRUE;
+ mark = TRUE;
+ }
+#ifdef MOTION_MONITOR
+ if (xtext->hilight_ent == ent &&
+ xtext->hilight_start <= i + offset && xtext->hilight_end > i + offset)
+ {
+ xtext->underline = TRUE;
+/* xtext->bold = TRUE;*/
+ hilight = TRUE;
+ }
+#endif
+
+ if (!xtext->double_buffer)
+ {
+ /* draw background to the left of the text */
+ if (str == ent->str && indent && xtext->time_stamp)
+ {
+ /* don't overwrite the timestamp */
+ if (indent > xtext->stamp_width)
+ {
+ if (!xtext->skip_border_fills)
+ gdk_draw_rectangle (xtext->draw_buf, xtext->bgc, 1,
+ xtext->stamp_width, y - xtext->font->ascent,
+ indent - xtext->stamp_width, xtext->fontsize);
+ }
+ } else
+ {
+ /* fill the indent area with background gc */
+ if (!xtext->skip_border_fills)
+ gdk_draw_rectangle (xtext->draw_buf, xtext->bgc, 1,
+ 0, y - xtext->font->ascent, indent, xtext->fontsize);
+ }
+ }
+
+ while (i < len)
+ {
+
+#ifdef MOTION_MONITOR
+ if (xtext->hilight_ent == ent && xtext->hilight_start == (i + offset))
+ {
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j;
+ j = 0;
+ xtext->underline = TRUE;
+/* xtext->bold = TRUE;*/
+ hilight = TRUE;
+ }
+#endif
+
+ if (!mark && ent->mark_start == (i + offset))
+ {
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j;
+ j = 0;
+ xtext_set_bg (gc, xtext->palette[16]);
+ xtext_set_fg (gc, xtext->palette[17]);
+ xtext->backcolor = TRUE;
+ mark = TRUE;
+ }
+
+ if ((xtext->parsing_color && isdigit (str[i]) && xtext->nc < 2) ||
+ (xtext->parsing_color && str[i] == ',' && xtext->nc < 3))
+ {
+ pstr++;
+ if (str[i] == ',')
+ {
+ xtext->parsing_backcolor = TRUE;
+ if (xtext->nc)
+ {
+ xtext->num[xtext->nc] = 0;
+ xtext->nc = 0;
+ col_num = atoi (xtext->num) % 16;
+ xtext->col_fore = col_num;
+ if (!mark)
+ xtext_set_fg (gc, xtext->palette[col_num]);
+ }
+ } else
+ {
+ xtext->num[xtext->nc] = str[i];
+ if (xtext->nc < 7)
+ xtext->nc++;
+ }
+ } else
+ {
+ if (xtext->parsing_color)
+ {
+ xtext->parsing_color = FALSE;
+ if (xtext->nc)
+ {
+ xtext->num[xtext->nc] = 0;
+ xtext->nc = 0;
+ col_num = atoi (xtext->num);
+ if (col_num == 99) /* mIRC lameness */
+ col_num = 19;
+ else
+ col_num = col_num % 16;
+ if (xtext->parsing_backcolor)
+ {
+ if (col_num == 1)
+ xtext->backcolor = FALSE;
+ else
+ xtext->backcolor = TRUE;
+ if (!mark)
+ xtext_set_bg (gc, xtext->palette[col_num]);
+ xtext->col_back = col_num;
+ } else
+ {
+ if (!mark)
+ xtext_set_fg (gc, xtext->palette[col_num]);
+ xtext->col_fore = col_num;
+ }
+ xtext->parsing_backcolor = FALSE;
+ } else
+ {
+ /* got a \003<non-digit>... i.e. reset colors */
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j;
+ j = 0;
+ gtk_xtext_reset (xtext, mark, FALSE);
+ }
+ }
+
+ switch (str[i])
+ {
+ case '\t':
+ str[i] = ' ';
+ j++;
+ break;
+ case '\n':
+ case ATTR_BEEP:
+ break;
+ case ATTR_REVERSE:
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j + 1;
+ j = 0;
+ tmp = xtext->col_fore;
+ xtext->col_fore = xtext->col_back;
+ xtext->col_back = tmp;
+ if (!mark)
+ {
+ xtext_set_fg (gc, xtext->palette[xtext->col_fore]);
+ xtext_set_bg (gc, xtext->palette[xtext->col_back]);
+ }
+ if (xtext->col_back != 19)
+ xtext->backcolor = TRUE;
+ else
+ xtext->backcolor = FALSE;
+ break;
+ case ATTR_BOLD:
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ xtext->bold = !xtext->bold;
+ pstr += j + 1;
+ j = 0;
+ break;
+ case ATTR_UNDERLINE:
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ xtext->underline = !xtext->underline;
+ pstr += j + 1;
+ j = 0;
+ break;
+ case ATTR_RESET:
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j + 1;
+ j = 0;
+ gtk_xtext_reset (xtext, mark, !hilight);
+ break;
+ case ATTR_COLOR:
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ xtext->parsing_color = TRUE;
+ pstr += j + 1;
+ j = 0;
+ break;
+ default:
+ j++;
+ }
+ }
+ i++;
+
+#ifdef MOTION_MONITOR
+ if (xtext->hilight_ent == ent && xtext->hilight_end == (i + offset))
+ {
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j;
+ j = 0;
+ xtext->underline = FALSE;
+/* xtext->bold = FALSE;*/
+ hilight = FALSE;
+ }
+#endif
+
+ if (mark && ent->mark_end == (i + offset))
+ {
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+ pstr += j;
+ j = 0;
+ xtext_set_bg (gc, xtext->palette[xtext->col_back]);
+ xtext_set_fg (gc, xtext->palette[xtext->col_fore]);
+ if (xtext->col_back != 19)
+ xtext->backcolor = TRUE;
+ else
+ xtext->backcolor = FALSE;
+ mark = FALSE;
+ }
+ }
+
+ if (j)
+ x += gtk_xtext_render_flush (xtext, x, y, pstr, j, gc);
+
+ if (!xtext->double_buffer)
+ {
+ /* draw separator now so it doesn't appear to flicker */
+ gtk_xtext_draw_sep (xtext, y + 1);
+ /* draw background to the right of the text */
+ if (!xtext->skip_border_fills)
+ gdk_draw_rectangle (xtext->draw_buf, xtext->bgc, 1,
+ x, y - xtext->font->ascent, (win_width + MARGIN) - x,
+ xtext->fontsize);
+ }
+}
+
+#ifdef USE_XLIB
+
+/* get the desktop/root window - thanks Eterm */
+
+static Window desktop_window = None;
+
+Window
+get_desktop_window (Window the_window)
+{
+ Atom prop, type, prop2;
+ int format;
+ unsigned long length, after;
+ unsigned char *data;
+ unsigned int nchildren;
+ Window w, root, *children, parent;
+
+ prop = XInternAtom (GDK_DISPLAY (), "_XROOTPMAP_ID", True);
+ prop2 = XInternAtom (GDK_DISPLAY (), "_XROOTCOLOR_PIXEL", True);
+
+ if (prop == None && prop2 == None)
+ return None;
+
+ for (w = the_window; w; w = parent)
+ {
+ if ((XQueryTree (GDK_DISPLAY (), w, &root, &parent, &children,
+ &nchildren)) == False)
+ return None;
+
+ if (nchildren)
+ XFree (children);
+
+ if (prop != None)
+ {
+ XGetWindowProperty (GDK_DISPLAY (), w, prop, 0L, 1L, False,
+ AnyPropertyType, &type, &format, &length, &after,
+ &data);
+ } else
+ {
+ XGetWindowProperty (GDK_DISPLAY (), w, prop2, 0L, 1L, False,
+ AnyPropertyType, &type, &format, &length, &after,
+ &data);
+ }
+
+ if (data)
+ XFree (data);
+
+ if (type != None)
+ {
+ return (desktop_window = w);
+ }
+ }
+
+ return (desktop_window = None);
+}
+
+/* stolen from zvt, which was stolen from Eterm */
+
+static Pixmap
+get_pixmap_prop (Window the_window)
+{
+ Atom prop, type;
+ int format;
+ unsigned long length, after;
+ unsigned char *data;
+ Pixmap pix = None;
+
+ if (desktop_window == None)
+ desktop_window = get_desktop_window (the_window);
+ if (desktop_window == None)
+ desktop_window = GDK_ROOT_WINDOW ();
+
+ prop = XInternAtom (GDK_DISPLAY (), "_XROOTPMAP_ID", True);
+ if (prop == None)
+ return None;
+
+ XGetWindowProperty (GDK_DISPLAY (), desktop_window, prop, 0L, 1L, False,
+ AnyPropertyType, &type, &format, &length, &after,
+ &data);
+ if (data)
+ {
+ if (type == XA_PIXMAP)
+ pix = *((Pixmap *) data);
+
+ XFree (data);
+ }
+
+ return pix;
+}
+
+#ifdef USE_GDK_PIXBUF
+
+static GdkPixmap *
+create_shaded_pixmap (GtkXText * xtext, Pixmap p, int x, int y, int w, int h)
+{
+ GdkPixmap *pp, *tmp, *shaded_pixmap;
+ GdkPixbuf *pixbuf;
+ GdkColormap *cmap;
+ GdkGC *tgc;
+ unsigned char *buf;
+ unsigned char *pbuf;
+ int width, height, depth;
+ int rowstride;
+ int pbwidth;
+ int pbheight;
+ int i, j;
+ int offset;
+ int r, g, b, a;
+
+ pp = gdk_pixmap_foreign_new (p);
+ cmap = gtk_widget_get_colormap (GTK_WIDGET (xtext));
+ gdk_window_get_geometry (pp, NULL, NULL, &width, &height, &depth);
+
+ if (width < x + w || height < y + h || x < 0 || y < 0)
+ {
+ tgc = gdk_gc_new (pp);
+ tmp = gdk_pixmap_new (pp, w, h, depth);
+ gdk_gc_set_tile (tgc, pp);
+ gdk_gc_set_fill (tgc, GDK_TILED);
+ gdk_gc_set_ts_origin (tgc, -x, -y);
+ gdk_draw_rectangle (tmp, tgc, TRUE, 0, 0, w, h);
+ gdk_gc_destroy (tgc);
+
+ pixbuf = gdk_pixbuf_get_from_drawable (NULL, tmp, cmap,
+ 0, 0, 0, 0, w, h);
+ gdk_pixmap_unref (tmp);
+ } else
+ {
+ pixbuf = gdk_pixbuf_get_from_drawable (NULL, pp, cmap,
+ x, y, 0, 0, w, h);
+ }
+ gdk_xid_table_remove (GDK_WINDOW_XWINDOW (pp));
+ g_dataset_destroy (pp);
+ g_free (pp);
+
+ if (!pixbuf)
+ return NULL;
+
+ buf = gdk_pixbuf_get_pixels (pixbuf);
+ rowstride = gdk_pixbuf_get_rowstride (pixbuf);
+ pbwidth = gdk_pixbuf_get_width (pixbuf);
+ pbheight = gdk_pixbuf_get_height (pixbuf);
+
+ a = 128; /* alpha */
+ r = xtext->tint_red;
+ g = xtext->tint_green;
+ b = xtext->tint_blue;
+
+ if (gdk_pixbuf_get_has_alpha (pixbuf))
+ offset = 4;
+ else
+ offset = 3;
+
+ for (i=0;i<pbheight;i++)
+ {
+ pbuf = buf;
+ for (j=0;j<pbwidth;j++)
+ {
+ pbuf[0] = ((pbuf[0] * r) >> 8);
+ pbuf[1] = ((pbuf[1] * g) >> 8);
+ pbuf[2] = ((pbuf[2] * b) >> 8);
+ pbuf+=offset;
+ }
+ buf+=rowstride;
+ }
+
+ gdk_pixbuf_render_pixmap_and_mask (pixbuf, &shaded_pixmap, NULL, 0);
+ gdk_pixbuf_unref (pixbuf);
+
+ return shaded_pixmap;
+}
+
+#endif
+
+/* free transparency xtext->pixmap */
+
+static void
+gtk_xtext_free_trans (GtkXText * xtext)
+{
+ if (xtext->pixmap)
+ {
+ if (xtext->shaded)
+ {
+ gdk_pixmap_unref (xtext->pixmap);
+ } else
+ {
+ gdk_xid_table_remove (GDK_WINDOW_XWINDOW (xtext->pixmap));
+ g_dataset_destroy (xtext->pixmap);
+ g_free (xtext->pixmap);
+ }
+ xtext->pixmap = NULL;
+ }
+}
+
+/* grab pixmap from root window and set xtext->pixmap */
+
+static void
+gtk_xtext_load_trans (GtkXText * xtext)
+{
+ Pixmap rootpix;
+ Window childret;
+ GtkWidget *widget = GTK_WIDGET (xtext);
+ int x, y;
+
+ rootpix = get_pixmap_prop (GDK_WINDOW_XWINDOW (widget->window));
+ if (rootpix == None)
+ {
+ if (xtext->error_function)
+ xtext->error_function ("Unable to get root window pixmap!\n\n"
+ "You may need to use Esetroot or Gnome\n"
+ "control-center to set your background.\n");
+ xtext->transparent = FALSE;
+ return;
+ }
+
+ XTranslateCoordinates (GDK_WINDOW_XDISPLAY (widget->window),
+ GDK_WINDOW_XWINDOW (widget->window),
+ GDK_ROOT_WINDOW (), 0, 0, &x, &y, &childret);
+
+#ifdef USE_GDK_PIXBUF
+ if (xtext->shaded)
+ {
+ int width, height;
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, &width, &height);
+ xtext->pixmap =
+ create_shaded_pixmap (xtext, rootpix, x, y, width, height);
+ gdk_gc_set_tile (xtext->bgc, xtext->pixmap);
+ gdk_gc_set_ts_origin (xtext->bgc, 0, 0);
+ } else
+ {
+#endif
+ xtext->pixmap = gdk_pixmap_foreign_new (rootpix);
+ gdk_gc_set_tile (xtext->bgc, xtext->pixmap);
+ gdk_gc_set_ts_origin (xtext->bgc, -x, -y);
+#ifdef USE_GDK_PIXBUF
+ }
+#endif
+ gdk_gc_set_fill (xtext->bgc, GDK_TILED);
+}
+
+#endif
+
+/* render a single line, which may wrap to more lines */
+
+static int
+gtk_xtext_render_line (GtkXText * xtext, textentry * ent, char *str, int len,
+ int line, int lines_max, int subline, int indent,
+ int str_width)
+{
+ char *time_str;
+ int y;
+ int width;
+ int orig_len;
+ int ret = 1;
+ int tmp;
+
+ if (!str)
+ return 0;
+
+ if (len == -1)
+ len = strlen (str);
+ orig_len = len;
+
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, &width, 0);
+ width -= MARGIN;
+
+ if (xtext->time_stamp)
+ {
+ time_str = ctime (&ent->stamp) + 10;
+ time_str[0] = '[';
+ time_str[9] = ']';
+ time_str[10] = 0;
+ y = (xtext->fontsize * line) + xtext->font->ascent;
+ gtk_xtext_render_str (xtext, y, ent, time_str, 10, width, 2, line);
+ }
+
+ startrl:
+
+ y = (xtext->fontsize * line) + xtext->font->ascent;
+
+ if (str_width == -1)
+ str_width = gtk_xtext_text_width (xtext, str, len);
+
+ str_width += indent;
+
+ tmp = 0;
+ while (str_width > width || (!is_del (str[len]) && xtext->wordwrap))
+ {
+ if (str_width <= width && !tmp)
+ tmp = len;
+ len--;
+ if (xtext->wordwrap && tmp - len > WORDWRAP_LIMIT)
+ {
+ len = tmp;
+ str_width = gtk_xtext_text_width (xtext, str, len) + indent;
+ break;
+ }
+ if (len == 0)
+ return 1;
+
+ /* this is quite a HACK but it speeds things up! */
+ if (str_width > width + 256)
+ len -= 10;
+ /* -- */
+
+ str_width = gtk_xtext_text_width (xtext, str, len) + indent;
+ }
+
+ if (!subline)
+ {
+ gtk_xtext_render_str (xtext, y, ent, str, len, width, indent, line);
+ } else
+ {
+ xtext->dont_render = TRUE;
+ gtk_xtext_render_str (xtext, y, ent, str, len, width, indent, line);
+ xtext->dont_render = FALSE;
+ subline--;
+ line--;
+ ret--;
+ }
+
+ if (xtext->wordwrap && str[len] == ' ')
+ len++;
+
+ if (len != orig_len && lines_max > line + 1)
+ { /* FIXME: recursion sux! */
+/* ret += gtk_xtext_render_line (xtext, ent, str + len, -1, line+1, lines_max, subline, xtext->indent, -1);*/
+ ret++;
+ str += len;
+ len = orig_len = strlen (str);
+ line++;
+ indent = xtext->indent;
+ str_width = -1;
+ /* FIXME: gotos suck! */
+ goto startrl;
+ }
+
+ return ret;
+}
+
+void
+gtk_xtext_set_palette (GtkXText * xtext, GdkColor palette[])
+{
+ int i;
+
+ for (i = 0; i < 20; i++)
+ xtext->palette[i] = palette[i].pixel;
+
+ if (GTK_WIDGET_REALIZED (xtext))
+ {
+ xtext_set_fg (xtext->fgc, xtext->palette[18]);
+ xtext_set_bg (xtext->fgc, xtext->palette[19]);
+ xtext_set_fg (xtext->bgc, xtext->palette[19]);
+ }
+ xtext->col_fore = 18;
+ xtext->col_back = 19;
+}
+
+static void
+gtk_xtext_fix_indent (GtkXText * xtext)
+{
+ int j;
+
+ if (xtext->indent) /* make indent a multiple of the space width */
+ {
+ j = 0;
+ while (j < xtext->indent)
+ {
+ j += xtext->space_width;
+ }
+ xtext->indent = j;
+ }
+}
+
+static void
+gtk_xtext_recalc_widths (GtkXText * xtext, int do_str_width)
+{
+ textentry *ent;
+
+ /* since we have a new font, we have to recalc the text widths */
+ ent = xtext->text_first;
+ while (ent)
+ {
+ if (do_str_width)
+ {
+ ent->str_width =
+ gtk_xtext_text_width (xtext, ent->str, ent->str_len);
+ }
+ if (ent->left_len != -1)
+ {
+ ent->indent =
+ (xtext->indent -
+ gtk_xtext_text_width (xtext, ent->str,
+ ent->left_len)) - xtext->space_width;
+ if (ent->indent < MARGIN)
+ ent->indent = MARGIN;
+ }
+ ent = ent->next;
+ }
+
+ gtk_xtext_calc_lines (xtext, FALSE);
+}
+
+void
+gtk_xtext_set_font (GtkXText * xtext, GdkFont * font, char *name)
+{
+#ifdef USE_XLIB
+ unsigned char i;
+ XFontStruct *xfont;
+#endif
+
+ if (xtext->font)
+ gdk_font_unref (xtext->font);
+
+ if (font)
+ {
+ xtext->font = font;
+ gdk_font_ref (font);
+ } else
+ font = xtext->font = gdk_font_load (name);
+
+ if (!font)
+ font = xtext->font = gdk_font_load ("fixed");
+
+ switch (font->type)
+ {
+ case GDK_FONT_FONT:
+ xtext->fontsize = font->ascent + font->descent;
+#ifdef USE_XLIB
+ xfont = GDK_FONT_XFONT (font);
+ if ((xfont->min_byte1 == 0) && (xfont->max_byte1 == 0))
+ {
+ xtext->fonttype = FONT_1BYTE;
+ for (i = 0; i < 255; i++)
+ {
+ xtext->fontwidth[i] = gdk_char_width (font, i);
+ }
+ xtext->space_width = xtext->fontwidth[' '];
+ } else
+ {
+#endif
+ /* without X11 pretend they are all 2BYTE- This is ok, just
+ a bit slower. */
+ xtext->fonttype = FONT_2BYTE;
+ xtext->space_width = gdk_char_width (font, ' ');
+#ifdef USE_XLIB
+ }
+#endif
+ break;
+
+ case GDK_FONT_FONTSET:
+ xtext->fontsize = gdk_text_height (font, " ", 1);
+ xtext->fonttype = FONT_SET;
+ xtext->space_width = gdk_char_width (font, ' ');
+ break;
+ }
+
+#ifdef USE_XLIB
+ xfont = GDK_FONT_XFONT (font);
+ /* check if it's a fixed width font */
+ if (xfont->min_bounds.width == xfont->max_bounds.width)
+ xtext->fixed_width_font = TRUE;
+ else
+ xtext->fixed_width_font = FALSE;
+#else
+ /* kudgy fixed-width font checking */
+ if (xtext->space_width == gdk_char_width (xtext->font, 'Z'))
+ xtext->fixed_width_font = TRUE;
+ else
+ xtext->fixed_width_font = FALSE;
+#endif
+
+ xtext->stamp_width =
+ gtk_xtext_text_width (xtext, "[88:88:88]", 10) + MARGIN;
+
+ gtk_xtext_fix_indent (xtext);
+
+ if (GTK_WIDGET_REALIZED (xtext))
+ {
+ if (xtext->fonttype != FONT_SET)
+ gdk_gc_set_font (xtext->fgc, xtext->font);
+
+ gtk_xtext_recalc_widths (xtext, TRUE);
+ }
+}
+
+void
+gtk_xtext_set_background (GtkXText * xtext, GdkPixmap * pixmap, int trans,
+ int shaded)
+{
+ GdkGCValues val;
+
+#ifndef USE_GDK_PIXBUF
+ shaded = FALSE;
+#endif
+
+#ifndef USE_XLIB
+ shaded = FALSE;
+ trans = FALSE;
+#endif
+
+ if (xtext->pixmap)
+ {
+#ifdef USE_XLIB
+ if (xtext->transparent)
+ gtk_xtext_free_trans (xtext);
+ else
+#endif
+ gdk_pixmap_unref (xtext->pixmap);
+ xtext->pixmap = NULL;
+ }
+
+ xtext->transparent = trans;
+
+#ifdef USE_XLIB
+ if (trans)
+ {
+ xtext->shaded = shaded;
+ if (GTK_WIDGET_REALIZED (xtext))
+ gtk_xtext_load_trans (xtext);
+ return;
+ }
+#endif
+
+ xtext->pixmap = pixmap;
+
+ if (pixmap != 0)
+ {
+ gdk_pixmap_ref (pixmap);
+ if (GTK_WIDGET_REALIZED (xtext))
+ {
+ gdk_gc_set_tile (xtext->bgc, pixmap);
+ gdk_gc_set_ts_origin (xtext->bgc, 0, 0);
+ gdk_gc_set_fill (xtext->bgc, GDK_TILED);
+ }
+ } else
+ {
+ if (GTK_WIDGET_REALIZED (xtext))
+ {
+ gdk_gc_destroy (xtext->bgc);
+ val.subwindow_mode = GDK_INCLUDE_INFERIORS;
+ val.graphics_exposures = 0;
+ xtext->bgc = gdk_gc_new_with_values (GTK_WIDGET (xtext)->window,
+ &val, GDK_GC_EXPOSURES | GDK_GC_SUBWINDOW);
+ xtext_set_fg (xtext->bgc, xtext->palette[19]);
+ }
+ }
+}
+
+gchar *
+gtk_xtext_get_chars (GtkXText * xtext)
+{
+ int lenght = 0;
+ gchar *chars;
+ textentry *tentry = xtext->text_first;
+ while (tentry != NULL)
+ {
+ lenght += tentry->str_len + 1;
+ tentry = tentry->next;
+ }
+ if (lenght == 0)
+ return NULL;
+ chars = g_malloc (lenght + 1);
+ *chars = 0;
+
+ tentry = xtext->text_first;
+ while (tentry != NULL)
+ {
+ strcat (chars, tentry->str);
+ strcat (chars, "\n");
+ tentry = tentry->next;
+ }
+
+ return chars;
+}
+
+static int
+gtk_xtext_lines_taken (GtkXText * xtext, textentry * ent)
+{
+ int tmp, orig_len, indent, len, win_width, str_width, lines = 0;
+ char *str;
+
+ str = ent->str;
+ len = orig_len = ent->str_len;
+ indent = ent->indent;
+
+ if (len < 2)
+ return 1;
+
+ win_width = GTK_WIDGET (xtext)->allocation.width - MARGIN;
+ str_width = ent->str_width + indent;
+
+ while (1)
+ {
+ lines++;
+ if (str_width <= win_width)
+ break;
+ tmp = 0;
+ while (str_width > win_width || (!is_del (str[len]) && xtext->wordwrap))
+ {
+ if (str_width <= win_width && !tmp)
+ tmp = len;
+ len--;
+ if (xtext->wordwrap && tmp - len > WORDWRAP_LIMIT)
+ {
+ len = tmp;
+ str_width = gtk_xtext_text_width (xtext, str, len) + indent;
+ break;
+ }
+ if (len == 0)
+ return 1;
+ if (str_width > win_width + 256) /* this might not work 100% but it */
+ len -= 10; /* sure speeds things up ALOT! */
+ str_width = gtk_xtext_text_width (xtext, str, len) + indent;
+ }
+
+ if (len == orig_len)
+ break;
+
+ if (xtext->wordwrap && str[len] == ' ')
+ len++;
+
+ str += len;
+ indent = xtext->indent;
+ len = strlen (str);
+ str_width = gtk_xtext_text_width (xtext, str, len) + indent;
+ }
+ return lines;
+}
+
+/* Calculate number of actual lines (with wraps), to set adj->lower. *
+ * This should only be called when the window resizes. */
+
+static void
+gtk_xtext_calc_lines (GtkXText * xtext, int fire_signal)
+{
+ textentry *ent;
+ int width;
+ int height;
+ int lines;
+
+ width = GTK_WIDGET (xtext)->allocation.width - MARGIN;
+ height = GTK_WIDGET (xtext)->allocation.height;
+
+ if (width < 30 || height < xtext->fontsize || width < xtext->indent + 30)
+ return;
+
+ lines = 0;
+ ent = xtext->text_first;
+ while (ent)
+ {
+ ent->lines_taken = gtk_xtext_lines_taken (xtext, ent);
+ lines += ent->lines_taken;
+ ent = ent->next;
+ }
+
+ xtext->pagetop_ent = NULL;
+ xtext->num_lines = lines;
+ gtk_xtext_adjustment_set (xtext, fire_signal);
+}
+
+/* find the n-th line in the linked list, this includes wrap calculations */
+
+static textentry *
+gtk_xtext_nth (GtkXText * xtext, textentry * ent, int line, int width,
+ int *subline)
+{
+ int lines = 0;
+
+ if (ent == NULL)
+ {
+ ent = xtext->text_first;
+ line += xtext->adj->value;
+ } else
+ {
+ lines -= *subline;
+ }
+
+ while (ent)
+ {
+ lines += ent->lines_taken;
+ if (lines > line)
+ {
+ *subline = ent->lines_taken - (lines - line);
+ return ent;
+ }
+ ent = ent->next;
+ }
+ return 0;
+}
+
+static void
+gtk_xtext_draw_sep (GtkXText * xtext, int y)
+{
+ int x, height;
+ GdkGC *light, *dark;
+
+ if (y == -1)
+ {
+ y = 2;
+ height = GTK_WIDGET (xtext)->allocation.height - 2;
+ } else
+ {
+ height = xtext->fontsize;
+ }
+
+ /* draw the separator line */
+ if (xtext->separator && xtext->indent)
+ {
+ light = xtext->light_gc;
+ dark = xtext->dark_gc;
+
+ x = xtext->indent - ((xtext->space_width + 1) / 2);
+ if (x < 1)
+ return;
+
+ if (xtext->thinline)
+ {
+ if (xtext->moving_separator)
+ gdk_draw_line (xtext->draw_buf, light, x, y, x, height);
+ else
+ gdk_draw_line (xtext->draw_buf, dark, x, y, x, height);
+ } else
+ {
+ if (xtext->moving_separator)
+ {
+ gdk_draw_line (xtext->draw_buf, light, x - 1, y, x - 1, height);
+ gdk_draw_line (xtext->draw_buf, dark, x, y, x, height);
+ } else
+ {
+ gdk_draw_line (xtext->draw_buf, dark, x - 1, y, x - 1, height);
+ gdk_draw_line (xtext->draw_buf, light, x, y, x, height);
+ }
+ }
+ }
+}
+
+/* render 2 ents (or an inclusive range) */
+
+static void
+gtk_xtext_render_ents (GtkXText * xtext, textentry * enta, textentry * entb,
+ int inclusive)
+{
+ textentry *ent, *orig_ent, *tmp_ent;
+ int line;
+ int lines_taken;
+ int lines_max;
+ int width;
+ int height;
+ int subline;
+ int drawing = FALSE;
+
+ if (xtext->double_buffer)
+ {
+ gtk_xtext_render_page (xtext);
+ return;
+ }
+
+ if (xtext->indent < MARGIN)
+ xtext->indent = MARGIN; /* 2 pixels is our left margin */
+
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, &width, &height);
+ width -= MARGIN;
+
+ if (width < 32 || height < xtext->fontsize || width < xtext->indent + 30)
+ return;
+
+ lines_max = (height - xtext->font->descent) / xtext->fontsize;
+ line = 0;
+ orig_ent = xtext->pagetop_ent;
+ subline = xtext->pagetop_subline;
+
+ /* check if enta is before the start of this page */
+ if (inclusive)
+ {
+ tmp_ent = orig_ent;
+ while (tmp_ent)
+ {
+ if (tmp_ent == enta)
+ break;
+ if (tmp_ent == entb)
+ {
+ drawing = TRUE;
+ break;
+ }
+ tmp_ent = tmp_ent->next;
+ }
+ }
+
+ line = 0;
+
+ ent = orig_ent;
+ while (ent)
+ {
+ if (inclusive && ent == enta)
+ drawing = TRUE;
+
+ if (drawing || ent == entb || ent == enta)
+ {
+ gtk_xtext_reset (xtext, FALSE, TRUE);
+ lines_taken = gtk_xtext_render_line (xtext, ent, ent->str,
+ ent->str_len, line, lines_max,
+ subline, ent->indent,
+ ent->str_width);
+ line += ent->lines_taken;
+ line -= subline;
+ if (ent == orig_ent)
+ subline = 0;
+ } else
+ {
+ if (ent == orig_ent)
+ {
+ line -= subline;
+ subline = 0;
+ }
+ line += ent->lines_taken;
+ }
+
+ if (inclusive && ent == entb)
+ break;
+
+ if (line >= lines_max)
+ break;
+
+ ent = ent->next;
+ }
+
+ /* draw the separator line */
+ gtk_xtext_draw_sep (xtext, -1);
+}
+
+/* render a whole page/window, starting from 'startline' */
+
+static void
+gtk_xtext_render_page (GtkXText * xtext)
+{
+ textentry *ent;
+ int line;
+ int lines_max;
+ int width;
+ int height;
+ int subline;
+ int startline = xtext->adj->value;
+
+ if (xtext->indent < MARGIN)
+ xtext->indent = MARGIN; /* 2 pixels is our left margin */
+
+ gdk_window_get_size (GTK_WIDGET (xtext)->window, &width, &height);
+ width -= MARGIN;
+
+ if (width < 32 || height < xtext->fontsize || width < xtext->indent + 30)
+ return;
+
+ lines_max = (height - xtext->font->descent) / xtext->fontsize;
+
+ subline = line = 0;
+ ent = xtext->text_first;
+
+ if (startline > 0)
+ ent = gtk_xtext_nth (xtext, ent, startline, width, &subline);
+
+ xtext->pagetop_ent = ent;
+ xtext->pagetop_subline = subline;
+
+ if (xtext->double_buffer)
+ {
+ xtext->tmp_pix = gdk_pixmap_new (((GtkWidget*)xtext)->window,
+ width + MARGIN, height, xtext->depth);
+ xtext->draw_buf = xtext->tmp_pix;
+ /* render the backdrop */
+ gdk_draw_rectangle (xtext->draw_buf, xtext->bgc, 1, 0, 0,
+ width + MARGIN, height);
+ }
+
+ while (ent)
+ {
+ gtk_xtext_reset (xtext, FALSE, TRUE);
+ line +=
+ gtk_xtext_render_line (xtext, ent, ent->str, ent->str_len, line,
+ lines_max, subline, ent->indent,
+ ent->str_width);
+ subline = 0;
+
+ if (line >= lines_max)
+ break;
+
+ ent = ent->next;
+ }
+
+ if (!xtext->double_buffer)
+ {
+ line = (xtext->fontsize * line);
+ /* fill any space below the last line with our background GC */
+ gdk_draw_rectangle (xtext->draw_buf, xtext->bgc, 1,
+ 0, line, width + MARGIN, height - line);
+ }
+
+ /* draw the separator line */
+ gtk_xtext_draw_sep (xtext, -1);
+
+ if (xtext->double_buffer)
+ {
+ /* send our double buffer to the actual window */
+ gdk_draw_pixmap (((GtkWidget*)xtext)->window, xtext->fgc, xtext->tmp_pix,
+ 0, 0, 0, 0, width + MARGIN, height);
+ gdk_pixmap_unref (xtext->tmp_pix);
+ }
+}
+
+void
+gtk_xtext_refresh (GtkXText * xtext, int do_trans)
+{
+ if (GTK_WIDGET_REALIZED (GTK_WIDGET (xtext)))
+ {
+#ifdef USE_XLIB
+ if (xtext->transparent && do_trans)
+ {
+ gtk_xtext_free_trans (xtext);
+ gtk_xtext_load_trans (xtext);
+ }
+#endif
+ gtk_xtext_render_page (xtext);
+ }
+}
+
+/* remove the topline from the list */
+
+static void
+gtk_xtext_remove_top (GtkXText * xtext)
+{
+ textentry *ent;
+
+ ent = xtext->text_first;
+ xtext->num_lines -= ent->lines_taken;
+ xtext->pagetop_ent = NULL;
+ xtext->text_first = ent->next;
+ free (ent);
+}
+
+void
+gtk_xtext_remove_lines (GtkXText * xtext, int lines, int refresh)
+{
+ textentry *next;
+
+ while (xtext->text_first && lines)
+ {
+ next = xtext->text_first->next;
+ free (xtext->text_first);
+ xtext->text_first = next;
+ lines--;
+ }
+ if (!xtext->text_first)
+ xtext->text_last = NULL;
+
+ if (refresh)
+ {
+ gtk_xtext_calc_lines (xtext, TRUE);
+ gtk_xtext_refresh (xtext, 0);
+ }
+}
+
+void *
+gtk_xtext_search (GtkXText * xtext, char *text, void *start)
+{
+ textentry *ent, *fent;
+ char *str;
+ int line;
+
+ gtk_xtext_selection_clear (xtext);
+
+ if (start)
+ ent = ((textentry *) start)->next;
+ else
+ ent = xtext->text_first;
+ while (ent)
+ {
+ if ((str = nocasestrstr (ent->str, text)))
+ {
+ ent->mark_start = str - ent->str;
+ ent->mark_end = ent->mark_start + strlen (text);
+ break;
+ }
+ ent = ent->next;
+ }
+
+ fent = ent;
+ ent = xtext->text_first;
+ line = 0;
+ while (ent)
+ {
+ line += ent->lines_taken;
+ ent = ent->next;
+ if (ent == fent)
+ break;
+ }
+ while (line > xtext->adj->upper - xtext->adj->page_size)
+ line--;
+
+ if (fent != 0)
+ {
+ xtext->adj->value = line;
+ xtext->scrollbar_down = FALSE;
+ gtk_adjustment_changed (xtext->adj);
+ }
+ gtk_xtext_render_page (xtext);
+
+ return fent;
+}
+
+static int
+gtk_xtext_render_page_timeout (GtkXText * xtext)
+{
+ GtkAdjustment *adj = xtext->adj;
+ gfloat val;
+
+ if (xtext->scrollbar_down)
+ {
+ gtk_xtext_adjustment_set (xtext, FALSE);
+ gtk_adjustment_set_value (adj, adj->upper - adj->page_size);
+ } else
+ {
+ val = adj->value;
+ gtk_xtext_adjustment_set (xtext, TRUE);
+ gtk_adjustment_set_value (adj, val);
+ }
+
+ if (adj->value >= adj->upper - adj->page_size || adj->value < 1)
+ gtk_xtext_render_page (xtext);
+
+ xtext->add_io_tag = -1;
+
+ return 0;
+}
+
+/* append a textentry to our linked list */
+
+static void
+gtk_xtext_append_entry (GtkXText * xtext, textentry * ent)
+{
+ ent->stamp = time (0);
+ ent->str_width = gtk_xtext_text_width (xtext, ent->str, ent->str_len);
+ ent->mark_start = -1;
+ ent->mark_end = -1;
+ ent->next = NULL;
+
+ if (ent->indent < MARGIN)
+ ent->indent = MARGIN; /* 2 pixels is the left margin */
+
+ /* append to our linked list */
+ if (xtext->text_last)
+ xtext->text_last->next = ent;
+ else
+ xtext->text_first = ent;
+ xtext->text_last = ent;
+
+ ent->lines_taken = gtk_xtext_lines_taken (xtext, ent);
+ xtext->num_lines += ent->lines_taken;
+
+ if (xtext->max_lines > 2 && xtext->max_lines < xtext->num_lines)
+ {
+ gtk_xtext_remove_top (xtext);
+ }
+
+/* if (xtext->frozen == 0 && xtext->add_io_tag == -1)*/
+ if (xtext->add_io_tag == -1)
+ {
+ xtext->add_io_tag = gtk_timeout_add (REFRESH_TIMEOUT * 2,
+ (GtkFunction)
+ gtk_xtext_render_page_timeout,
+ xtext);
+ }
+}
+
+/* the main two public functions */
+
+void
+gtk_xtext_append_indent (GtkXText * xtext,
+ char *left_text, int left_len,
+ char *right_text, int right_len)
+{
+ textentry *ent;
+ char *str;
+ int space;
+
+ if (left_len == -1)
+ left_len = strlen (left_text);
+
+ if (right_len == -1)
+ right_len = strlen (right_text);
+
+ ent = malloc (left_len + right_len + 2 + sizeof (textentry));
+ str = (char *) ent + sizeof (textentry);
+
+ space = xtext->indent - gtk_xtext_text_width (xtext, left_text, left_len);
+
+ memcpy (str, left_text, left_len);
+ str[left_len] = ' ';
+ memcpy (str + left_len + 1, right_text, right_len);
+ str[left_len + 1 + right_len] = 0;
+
+ ent->left_len = left_len;
+ ent->str = str;
+ ent->str_len = left_len + 1 + right_len;
+ ent->indent = space - xtext->space_width;
+
+ if (xtext->time_stamp)
+ space = xtext->stamp_width;
+ else
+ space = 0;
+
+ /* do we need to auto adjust the separator position? */
+ if (xtext->auto_indent && ent->indent < MARGIN + space)
+ {
+ xtext->indent -= ent->indent;
+ xtext->indent += MARGIN;
+ xtext->indent += space;
+ if (xtext->indent > xtext->max_auto_indent)
+ xtext->indent = xtext->max_auto_indent;
+ gtk_xtext_fix_indent (xtext);
+ gtk_xtext_recalc_widths (xtext, FALSE);
+ space =
+ xtext->indent - gtk_xtext_text_width (xtext, left_text, left_len);
+ ent->indent = space - xtext->space_width;
+ }
+
+ gtk_xtext_append_entry (xtext, ent);
+}
+
+void
+gtk_xtext_append (GtkXText * xtext, char *text, int len)
+{
+ textentry *ent;
+
+ if (len == -1)
+ len = strlen (text);
+
+ ent = malloc (len + 1 + sizeof (textentry));
+ ent->str = (char *) ent + sizeof (textentry);
+ ent->str_len = len;
+ if (len)
+ memcpy (ent->str, text, len);
+ ent->str[len] = 0;
+ ent->indent = 0;
+ ent->left_len = -1;
+
+ gtk_xtext_append_entry (xtext, ent);
+}