medit/moo/mooterm/mootermbuffer.c
2005-06-22 18:20:32 +00:00

611 lines
20 KiB
C

/*
* mooterm/mootermbuffer.c
*
* Copyright (C) 2004-2005 by Yevgen Muntyan <muntyan@math.tamu.edu>
*
* 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.
*
* See COPYING file that comes with this distribution.
*/
#define MOOTERM_COMPILATION
#include "mooterm/mootermbuffer-private.h"
#include "mooutils/moocompat.h"
#include "mooutils/moomarshals.h"
#define MIN_WIDTH (10L)
#define MIN_HEIGHT (10L)
static void moo_term_buffer_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec);
static void moo_term_buffer_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec);
static GObject *moo_term_buffer_constructor (GType type,
guint n_construct_properties,
GObjectConstructParam *construct_param);
static void moo_term_buffer_finalize (GObject *object);
/* MOO_TYPE_TERM_BUFFER */
G_DEFINE_TYPE (MooTermBuffer, moo_term_buffer, G_TYPE_OBJECT)
enum {
CHANGED,
CURSOR_MOVED,
BELL,
FLASH_SCREEN,
SCREEN_SIZE_CHANGED,
LAST_SIGNAL
};
enum {
PROP_0,
PROP_SCREEN_WIDTH,
PROP_SCREEN_HEIGHT,
PROP_SCROLLBACK,
PROP_MAX_SCROLLBACK,
PROP_CURSOR_VISIBLE,
PROP_AM_MODE,
PROP_INSERT_MODE
};
static guint signals[LAST_SIGNAL];
static void moo_term_buffer_class_init (MooTermBufferClass *klass)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
gobject_class->set_property = moo_term_buffer_set_property;
gobject_class->get_property = moo_term_buffer_get_property;
gobject_class->constructor = moo_term_buffer_constructor;
gobject_class->finalize = moo_term_buffer_finalize;
klass->changed = buf_changed_clear;
klass->screen_size_changed = NULL;
klass->cursor_moved = NULL;
klass->bell = NULL;
klass->flash_screen = NULL;
signals[CHANGED] =
g_signal_new ("changed",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooTermBufferClass, changed),
NULL, NULL,
_moo_marshal_VOID__VOID,
G_TYPE_NONE, 0);
signals[CURSOR_MOVED] =
g_signal_new ("cursor-moved",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooTermBufferClass, cursor_moved),
NULL, NULL,
_moo_marshal_VOID__ULONG_ULONG,
G_TYPE_NONE, 2,
G_TYPE_ULONG, G_TYPE_ULONG);
signals[BELL] =
g_signal_new ("bell",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooTermBufferClass, bell),
NULL, NULL,
_moo_marshal_VOID__VOID,
G_TYPE_NONE, 0);
signals[FLASH_SCREEN] =
g_signal_new ("flash-screen",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooTermBufferClass, flash_screen),
NULL, NULL,
_moo_marshal_VOID__VOID,
G_TYPE_NONE, 0);
signals[SCREEN_SIZE_CHANGED] =
g_signal_new ("screen-size-changed",
G_OBJECT_CLASS_TYPE (gobject_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooTermBufferClass, screen_size_changed),
NULL, NULL,
_moo_marshal_VOID__ULONG_ULONG,
G_TYPE_NONE, 2,
G_TYPE_ULONG, G_TYPE_ULONG);
g_object_class_install_property (gobject_class,
PROP_SCREEN_WIDTH,
g_param_spec_ulong ("screen-width",
"screen-width",
"screen-width",
0, 1000000, 80,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
g_object_class_install_property (gobject_class,
PROP_SCREEN_HEIGHT,
g_param_spec_ulong ("screen-height",
"screen-height",
"screen-height",
0, 1000000, 24,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
g_object_class_install_property (gobject_class,
PROP_MAX_SCROLLBACK,
g_param_spec_long ("max-scrollback",
"max-scrollback",
"max-scrollback",
-1, 1000000, -1,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
g_object_class_install_property (gobject_class,
PROP_SCROLLBACK,
g_param_spec_ulong ("scrollback",
"scrollback",
"scrollback",
0, 1000000, 0,
G_PARAM_READABLE));
g_object_class_install_property (gobject_class,
PROP_CURSOR_VISIBLE,
g_param_spec_boolean ("cursor-visible",
"cursor-visible",
"cursor-visible",
TRUE,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
g_object_class_install_property (gobject_class,
PROP_AM_MODE,
g_param_spec_boolean ("am-mode",
"am-mode",
"am-mode",
TRUE,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
g_object_class_install_property (gobject_class,
PROP_INSERT_MODE,
g_param_spec_boolean ("insert-mode",
"insert-mode",
"insert-mode",
TRUE,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
}
static void moo_term_buffer_init (MooTermBuffer *buf)
{
buf->priv = g_new0 (MooTermBufferPrivate, 1);
buf->priv->lines = g_array_new (FALSE, FALSE, sizeof(MooTermLine));
buf->priv->changed = NULL;
buf->priv->parser = moo_term_parser_new (buf);
}
static void moo_term_buffer_finalize (GObject *object)
{
guint i;
MooTermBuffer *buf = MOO_TERM_BUFFER (object);
for (i = 0; i < buf->priv->lines->len; ++i)
term_line_destroy (buf_line (buf, i));
g_array_free (buf->priv->lines, TRUE);
if (buf->priv->changed)
buf_region_destroy (buf->priv->changed);
moo_term_parser_free (buf->priv->parser);
g_free (buf->priv);
G_OBJECT_CLASS (moo_term_buffer_parent_class)->finalize (object);
}
static void moo_term_buffer_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
MooTermBuffer *buf = MOO_TERM_BUFFER (object);
switch (prop_id) {
case PROP_SCREEN_WIDTH:
if (buf->priv->constructed)
moo_term_buffer_set_screen_size (buf, g_value_get_ulong (value), 0);
else
buf->priv->screen_width = g_value_get_ulong (value);
break;
case PROP_SCREEN_HEIGHT:
if (buf->priv->constructed)
moo_term_buffer_set_screen_size (buf, 0, g_value_get_ulong (value));
else
buf->priv->screen_height = g_value_get_ulong (value);
break;
case PROP_MAX_SCROLLBACK:
if (buf->priv->constructed)
moo_term_buffer_set_max_scrollback (buf, g_value_get_long (value));
else
buf->priv->max_height = g_value_get_long (value);
break;
case PROP_CURSOR_VISIBLE:
moo_term_buffer_set_cursor_visible (buf, g_value_get_boolean (value));
break;
case PROP_AM_MODE:
moo_term_buffer_set_am_mode (buf, g_value_get_boolean (value));
break;
case PROP_INSERT_MODE:
moo_term_buffer_set_insert_mode (buf, g_value_get_boolean (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void moo_term_buffer_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
MooTermBuffer *buf = MOO_TERM_BUFFER (object);
switch (prop_id) {
case PROP_SCREEN_WIDTH:
g_value_set_ulong (value, buf->priv->screen_width);
break;
case PROP_SCREEN_HEIGHT:
g_value_set_ulong (value, buf->priv->screen_height);
break;
case PROP_SCROLLBACK:
g_value_set_ulong (value, buf_screen_offset (buf));
break;
case PROP_MAX_SCROLLBACK:
g_value_set_long (value, buf->priv->max_height);
break;
case PROP_CURSOR_VISIBLE:
g_value_set_boolean (value, buf->priv->cursor_visible);
break;
case PROP_AM_MODE:
g_value_set_boolean (value, buf->priv->am_mode);
break;
case PROP_INSERT_MODE:
g_value_set_boolean (value, buf->priv->insert_mode);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static GObject *moo_term_buffer_constructor (GType type,
guint n_props,
GObjectConstructParam *props)
{
MooTermBuffer *buf;
GObject *object;
guint i;
object = G_OBJECT_CLASS(moo_term_buffer_parent_class)->constructor (type, n_props, props);
buf = MOO_TERM_BUFFER (object);
buf->priv->constructed = TRUE;
if (buf->priv->max_height <= MIN_HEIGHT)
buf->priv->max_height = -1;
buf->priv->current_attr.mask = 0;
if (buf->priv->screen_width < MIN_WIDTH)
buf->priv->screen_width = MIN_WIDTH;
if (buf->priv->screen_height < MIN_HEIGHT)
buf->priv->screen_height = MIN_HEIGHT;
buf->priv->screen_offset = 0;
for (i = 0; i < buf->priv->screen_height; ++i)
{
MooTermLine line;
term_line_init (&line, buf->priv->screen_width);
g_array_append_val (buf->priv->lines, line);
}
return object;
}
void moo_term_buffer_set_cursor_visible (MooTermBuffer *buf,
gboolean visible)
{
buf->priv->cursor_visible = visible;
g_object_notify (G_OBJECT (buf), "cursor-visible");
}
void moo_term_buffer_bell (MooTermBuffer *buf)
{
g_signal_emit (buf, signals[BELL], 0);
}
void moo_term_buffer_flash_screen (MooTermBuffer *buf)
{
g_signal_emit (buf, signals[FLASH_SCREEN], 0);
}
void moo_term_buffer_changed (MooTermBuffer *buf)
{
if (!buf->priv->freeze_changed_notify)
g_signal_emit (buf, signals[CHANGED], 0);
}
void moo_term_buffer_scrollback_changed (MooTermBuffer *buf)
{
g_object_notify (G_OBJECT (buf), "scrollback");
}
void moo_term_buffer_set_am_mode (MooTermBuffer *buf,
gboolean auto_margins)
{
buf->priv->am_mode = auto_margins;
g_object_notify (G_OBJECT (buf), "am-mode");
}
void moo_term_buffer_set_insert_mode (MooTermBuffer *buf,
gboolean insert_mode)
{
buf->priv->insert_mode = insert_mode;
g_object_notify (G_OBJECT (buf), "insert-mode");
}
void moo_term_buffer_set_screen_size (MooTermBuffer *buf,
gulong width,
gulong height)
{
gulong old_width = buf_screen_width (buf);
gulong old_height = buf_screen_height (buf);
guint i;
if (height >= MIN_HEIGHT && height != old_height)
{
buf->priv->screen_height = height;
if (height > old_height)
{
if (buf->priv->screen_offset < height - old_height)
{
guint add = height - old_height - buf->priv->screen_offset;
for (i = 0; i < add; ++i)
{
MooTermLine new_line;
term_line_init (&new_line, old_width);
g_array_append_val (buf->priv->lines, new_line);
}
if (buf->priv->screen_offset)
{
gulong cursor_row =
buf->priv->screen_offset + buf->priv->cursor_row;
buf->priv->screen_offset = 0;
buf_cursor_move_to (buf, cursor_row, -1);
moo_term_buffer_scrollback_changed (buf);
}
}
else
{
buf->priv->screen_offset -= (height - old_height);
buf_cursor_move_to (buf,
buf->priv->cursor_row + height - old_height,
-1);
moo_term_buffer_scrollback_changed (buf);
}
}
else
{
buf->priv->screen_offset += (old_height - height);
if (buf->priv->cursor_row > old_height - height)
buf_cursor_move_to (buf,
buf->priv->cursor_row + height - old_height,
-1);
else
buf_cursor_move_to (buf, 0, -1);
moo_term_buffer_scrollback_changed (buf);
}
g_object_notify (G_OBJECT (buf), "screen-height");
}
if (width >= MIN_WIDTH && width != old_width)
{
height = buf_screen_height (buf);
for (i = 0; i < height; ++i)
term_line_set_len (buf_screen_line (buf, i), width);
buf->priv->screen_width = width;
if (buf->priv->cursor_col >= width)
buf_cursor_move_to (buf, -1, width - 1);
g_object_notify (G_OBJECT (buf), "screen-width");
}
g_signal_emit (buf, signals[SCREEN_SIZE_CHANGED], 0,
buf_screen_width (buf),
buf_screen_height (buf));
buf_changed_set_all (buf);
moo_term_buffer_changed (buf);
}
void moo_term_buffer_set_max_scrollback (G_GNUC_UNUSED MooTermBuffer *buf,
G_GNUC_UNUSED glong lines)
{
g_warning ("%s: implement me", G_STRLOC);
}
void moo_term_buffer_cursor_move (MooTermBuffer *buf,
long rows,
long cols)
{
buf_cursor_move (buf, rows, cols);
}
static void moo_term_buffer_cursor_moved (MooTermBuffer *buf,
gulong old_row,
gulong old_col)
{
if (!buf->priv->freeze_cursor_notify)
g_signal_emit (buf, signals[CURSOR_MOVED], 0,
old_row, old_col);
}
void moo_term_buffer_cursor_move_to (MooTermBuffer *buf,
long row,
long col)
{
gulong old_row = buf_cursor_row (buf);
gulong old_col = buf_cursor_col (buf);
g_return_if_fail (row < (long)buf_screen_height (buf));
g_return_if_fail (col < (long)buf_screen_width (buf));
if (row < 0)
row = old_row;
if (col < 0)
col = old_col;
buf->priv->cursor_row = row;
buf->priv->cursor_col = col;
moo_term_buffer_cursor_moved (buf, old_row, old_col);
}
void moo_term_buffer_feed (MooTermBuffer *buf,
const char *data,
gssize len)
{
gulong old_cursor_row = buf_cursor_row (buf);
gulong old_cursor_col = buf_cursor_col (buf);
buf_freeze_changed_notify (buf);
buf_freeze_cursor_notify (buf);
moo_term_parser_parse (buf->priv->parser, data, len);
buf_thaw_changed_notify (buf);
buf_thaw_cursor_notify (buf);
moo_term_buffer_changed (buf);
moo_term_buffer_cursor_moved (buf, old_cursor_row, old_cursor_col);
}
static void buf_print_char (MooTermBuffer *buf,
char c)
{
gulong width = buf_screen_width (buf);
gulong cursor_row = buf_cursor_row (buf);
MooTermTextAttr *attr =
buf->priv->current_attr.mask ? &buf->priv->current_attr : NULL;
if (buf->priv->insert_mode)
{
term_line_insert_char (buf_screen_line (buf, cursor_row),
buf->priv->cursor_col++,
c, attr, width);
buf_changed_add_range (buf, cursor_row,
buf->priv->cursor_col - 1,
width - buf->priv->cursor_col + 1);
}
else
{
term_line_set_char (buf_screen_line (buf, cursor_row),
buf->priv->cursor_col++,
c, attr, width);
buf_changed_add_range (buf, cursor_row,
buf->priv->cursor_col - 1, 1);
}
if (buf->priv->cursor_col == width)
{
if (buf->priv->am_mode)
{
buf->priv->cursor_col = 0;
buf_line_feed (buf);
}
else
buf->priv->cursor_col--;
}
}
void moo_term_buffer_print_chars (MooTermBuffer *buf,
const char *chars,
gssize len)
{
gsize i;
if (!len)
return;
for (i = 0; (len < 0 && chars[i]) || (len > 0 && i < (gsize)len); ++i)
{
char c = chars[i];
if (c & '\200' || (' ' <= c && c <= '~'))
buf_print_char (buf, c);
else if (c == '\127')
buf_print_char (buf, '~');
else
{
buf_print_char (buf, '^');
buf_print_char (buf, c + 0100);
}
}
}
MooTermBuffer *moo_term_buffer_new (gulong width,
gulong height)
{
return MOO_TERM_BUFFER (g_object_new (MOO_TYPE_TERM_BUFFER,
"screen-width", width,
"screen-height", height,
NULL));
}