Mercurial > pidgin
view console/libgnt/gnttextview.c @ 14375:f597127368de
[gaim-migrate @ 17081]
Changing Hilbert to Hil, as per the patch submitter's request.
committer: Tailor Script <tailor@pidgin.im>
author | Richard Laager <rlaager@wiktel.com> |
---|---|
date | Tue, 29 Aug 2006 21:09:43 +0000 |
parents | f3137c1faebe |
children | 5f3058c7a7df |
line wrap: on
line source
#include "gnttextview.h" #include "gntutils.h" enum { SIGS = 1, }; typedef struct { GntTextFormatFlags tvflag; chtype flags; char *text; } GntTextSegment; typedef struct { GList *segments; /* A list of GntTextSegments */ int length; /* The current length of the line so far (ie. onscreen width) */ gboolean soft; /* TRUE if it's an overflow from prev. line */ } GntTextLine; static GntWidgetClass *parent_class = NULL; static guint signals[SIGS] = { 0 }; static void gnt_text_view_draw(GntWidget *widget) { GntTextView *view = GNT_TEXT_VIEW(widget); int i = 0; GList *lines; int rows, scrcol; werase(widget->window); for (i = 0, lines = view->list; i < widget->priv.height && lines; i++, lines = lines->next) { GList *iter; GntTextLine *line = lines->data; wmove(widget->window, widget->priv.height - 1 - i, 0); for (iter = line->segments; iter; iter = iter->next) { GntTextSegment *seg = iter->data; wattrset(widget->window, seg->flags); wprintw(widget->window, "%s", seg->text); if (!iter->next) whline(widget->window, ' ' | seg->flags, widget->priv.width - line->length - 1); } } scrcol = widget->priv.width - 1; rows = widget->priv.height - 2; if (rows > 0) { int total = g_list_length(g_list_first(view->list)); int showing, position, up, down; showing = rows * rows / total + 1; showing = MIN(rows, showing); total -= rows; up = g_list_length(lines); down = total - up; position = (rows - showing) * up / MAX(1, up + down); position = MAX((lines != NULL), position); if (showing + position > rows) position = rows - showing; if (showing + position == rows && view->list && view->list->prev) position = MAX(1, rows - 1 - showing); else if (showing + position < rows && view->list && !view->list->prev) position = rows - showing; mvwvline(widget->window, position + 1, scrcol, ACS_CKBOARD | COLOR_PAIR(GNT_COLOR_HIGHLIGHT_D), showing); } mvwaddch(widget->window, 0, scrcol, (lines ? ACS_UARROW : ' ') | COLOR_PAIR(GNT_COLOR_HIGHLIGHT_D)); mvwaddch(widget->window, widget->priv.height - 1, scrcol, ((view->list && view->list->prev) ? ACS_DARROW : ' ') | COLOR_PAIR(GNT_COLOR_HIGHLIGHT_D)); DEBUG; } static void gnt_text_view_size_request(GntWidget *widget) { if (!GNT_WIDGET_IS_FLAG_SET(widget, GNT_WIDGET_MAPPED)) { gnt_widget_set_size(widget, 64, 24); } } static void gnt_text_view_map(GntWidget *widget) { if (widget->priv.width == 0 || widget->priv.height == 0) gnt_widget_size_request(widget); DEBUG; } static gboolean gnt_text_view_key_pressed(GntWidget *widget, const char *text) { return FALSE; } static void free_text_segment(gpointer data, gpointer null) { GntTextSegment *seg = data; g_free(seg->text); g_free(seg); } static void free_text_line(gpointer data, gpointer null) { GntTextLine *line = data; g_list_foreach(line->segments, free_text_segment, NULL); g_list_free(line->segments); g_free(line); } static void gnt_text_view_destroy(GntWidget *widget) { GntTextView *view = GNT_TEXT_VIEW(widget); view->list = g_list_first(view->list); g_list_foreach(view->list, free_text_line, NULL); g_list_free(view->list); } static gboolean gnt_text_view_clicked(GntWidget *widget, GntMouseEvent event, int x, int y) { if (event == GNT_MOUSE_SCROLL_UP) { gnt_text_view_scroll(GNT_TEXT_VIEW(widget), -1); } else if (event == GNT_MOUSE_SCROLL_DOWN) { gnt_text_view_scroll(GNT_TEXT_VIEW(widget), 1); } else return FALSE; return TRUE; } static void gnt_text_view_reflow(GntTextView *view) { /* This is pretty ugly, and inefficient. Someone do something about it. */ GntTextLine *line; GList *back, *iter, *list; int pos = 0; list = view->list; while (list->prev) { line = list->data; if (!line->soft) pos++; list = list->prev; } back = g_list_last(view->list); view->list = NULL; gnt_text_view_clear(view); for (; back; back = back->prev) { line = back->data; if (back->next && !line->soft) gnt_text_view_next_line(view); for (iter = line->segments; iter; iter = iter->next) { GntTextSegment *seg = iter->data; gnt_text_view_append_text_with_flags(view, seg->text, seg->tvflag); } free_text_line(line, NULL); } g_list_free(list); list = view->list = g_list_first(view->list); while (pos--) { while (((GntTextLine*)list->data)->soft) list = list->next; list = list->next; } view->list = list; gnt_widget_draw(GNT_WIDGET(view)); } static void gnt_text_view_size_changed(GntWidget *widget, int w, int h) { if (w != widget->priv.width) { gnt_text_view_reflow(GNT_TEXT_VIEW(widget)); } } static void gnt_text_view_class_init(GntTextViewClass *klass) { parent_class = GNT_WIDGET_CLASS(klass); parent_class->destroy = gnt_text_view_destroy; parent_class->draw = gnt_text_view_draw; parent_class->map = gnt_text_view_map; parent_class->size_request = gnt_text_view_size_request; parent_class->key_pressed = gnt_text_view_key_pressed; parent_class->clicked = gnt_text_view_clicked; parent_class->size_changed = gnt_text_view_size_changed; DEBUG; } static void gnt_text_view_init(GTypeInstance *instance, gpointer class) { GntWidget *widget = GNT_WIDGET(instance); GNT_WIDGET_SET_FLAGS(GNT_WIDGET(instance), GNT_WIDGET_GROW_Y | GNT_WIDGET_GROW_X); widget->priv.minw = 5; widget->priv.minh = 2; DEBUG; } /****************************************************************************** * GntTextView API *****************************************************************************/ GType gnt_text_view_get_gtype(void) { static GType type = 0; if(type == 0) { static const GTypeInfo info = { sizeof(GntTextViewClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc)gnt_text_view_class_init, NULL, /* class_finalize */ NULL, /* class_data */ sizeof(GntTextView), 0, /* n_preallocs */ gnt_text_view_init, /* instance_init */ }; type = g_type_register_static(GNT_TYPE_WIDGET, "GntTextView", &info, 0); } return type; } GntWidget *gnt_text_view_new() { GntWidget *widget = g_object_new(GNT_TYPE_TEXTVIEW, NULL); GntTextView *view = GNT_TEXT_VIEW(widget); GntTextLine *line = g_new0(GntTextLine, 1); GNT_WIDGET_SET_FLAGS(widget, GNT_WIDGET_NO_BORDER | GNT_WIDGET_NO_SHADOW); view->list = g_list_append(view->list, line); return widget; } void gnt_text_view_append_text_with_flags(GntTextView *view, const char *text, GntTextFormatFlags flags) { GntWidget *widget = GNT_WIDGET(view); int fl = 0; char **split; int i; GList *list = view->list; if (text == NULL || *text == '\0') return; fl = gnt_text_format_flag_to_chtype(flags); view->list = g_list_first(view->list); split = g_strsplit(text, "\n", -1); for (i = 0; split[i]; i++) { GntTextLine *line; char *iter = split[i]; int prev = 0; if (i) { line = g_new0(GntTextLine, 1); view->list = g_list_prepend(g_list_first(view->list), line); } line = view->list->data; while (iter && *iter) { int len; len = gnt_util_onscreen_width_to_pointer(iter, widget->priv.width - line->length - 1, &prev) - iter; if (len) { GntTextSegment *seg = g_new0(GntTextSegment, 1); seg->flags = fl; seg->tvflag = flags; seg->text = g_new0(char, len + 1); g_utf8_strncpy(seg->text, iter, g_utf8_pointer_to_offset(iter, iter + len)); line->segments = g_list_append(line->segments, seg); line->length += prev; iter += len; if (line->length >= widget->priv.width - 1 && *iter) { line = g_new0(GntTextLine, 1); line->soft = TRUE; view->list = g_list_prepend(g_list_first(view->list), line); } } else { line = g_new0(GntTextLine, 1); line->soft = TRUE; view->list = g_list_prepend(g_list_first(view->list), line); } } } g_strfreev(split); view->list = list; gnt_widget_draw(widget); } void gnt_text_view_scroll(GntTextView *view, int scroll) { if (scroll == 0) { view->list = g_list_first(view->list); } else if (scroll > 0) { GList *list = g_list_nth_prev(view->list, scroll); if (list == NULL) list = g_list_first(view->list); view->list = list; } else if (scroll < 0) { GList *list = g_list_nth(view->list, -scroll); if (list == NULL) list = g_list_last(view->list); view->list = list; } gnt_widget_draw(GNT_WIDGET(view)); } void gnt_text_view_next_line(GntTextView *view) { GntTextLine *line = g_new0(GntTextLine, 1); GList *list = view->list; view->list = g_list_prepend(g_list_first(view->list), line); view->list = list; gnt_widget_draw(GNT_WIDGET(view)); } chtype gnt_text_format_flag_to_chtype(GntTextFormatFlags flags) { chtype fl = 0; if (flags & GNT_TEXT_FLAG_BOLD) fl |= A_BOLD; if (flags & GNT_TEXT_FLAG_UNDERLINE) fl |= A_UNDERLINE; if (flags & GNT_TEXT_FLAG_BLINK) fl |= A_BLINK; if (flags & GNT_TEXT_FLAG_DIM) fl |= (A_DIM | COLOR_PAIR(GNT_COLOR_DISABLED)); else if (flags & GNT_TEXT_FLAG_HIGHLIGHT) fl |= (A_DIM | COLOR_PAIR(GNT_COLOR_HIGHLIGHT)); else fl |= COLOR_PAIR(GNT_COLOR_NORMAL); return fl; } void gnt_text_view_clear(GntTextView *view) { GntTextLine *line; g_list_foreach(view->list, free_text_line, NULL); g_list_free(view->list); view->list = NULL; line = g_new0(GntTextLine, 1); view->list = g_list_append(view->list, line); if (GNT_WIDGET(view)->window) gnt_widget_draw(GNT_WIDGET(view)); } int gnt_text_view_get_lines_below(GntTextView *view) { int below = 0; GList *list = view->list; while ((list = list->prev)) ++below; return below; } int gnt_text_view_get_lines_above(GntTextView *view) { int above = 0; GList *list = view->list; list = g_list_nth(view->list, GNT_WIDGET(view)->priv.height); if (!list) return 0; while ((list = list->next)) ++above; return above; }