medit/moo/mooedit/moohighlighter.c
2006-05-27 20:55:06 -05:00

814 lines
22 KiB
C

/*
* moohighlighter.c
*
* Copyright (C) 2004-2006 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 MOOEDIT_COMPILATION
#include "mooedit/moohighlighter.h"
#include "mooedit/moolang-rules.h"
#include "mooedit/gtksourceiter.h"
#include "mooedit/mootext-private.h"
#include <string.h>
#define IDLE_HIGHLIGHT_PRIORITY GTK_TEXT_VIEW_PRIORITY_VALIDATE
#define IDLE_HIGHLIGHT_TIME 30
#define COMPUTE_NOW_TIME 20
static MooSyntaxTag *iter_get_syntax_tag (const GtkTextIter *iter);
static GtkTextTag *moo_syntax_tag_new (CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule);
static GtkTextTag *create_tag (MooHighlighter *hl,
CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule);
static GtkTextTag *get_syntax_tag (MooHighlighter *hl,
CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule);
static CtxNode *ctx_node_new (MooHighlighter *hl,
CtxNode *parent,
MooContext *context,
gboolean create_tag);
static void ctx_node_free (CtxNode *node,
GtkTextTagTable *table);
static CtxNode *get_next_node (MooHighlighter *hl,
CtxNode *node,
MooRule *rule);
static CtxNode *get_line_end_node (MooHighlighter *hl,
CtxNode *node);
static CtxNode *hl_compute_line (MooHighlighter *hl,
Line *line,
MatchData *data,
CtxNode *node);
static gboolean hl_compute_range (MooHighlighter *hl,
Interval *lines,
int time);
/* MOO_TYPE_SYNTAX_TAG */
G_DEFINE_TYPE (MooSyntaxTag, _moo_syntax_tag, GTK_TYPE_TEXT_TAG)
static void
_moo_syntax_tag_class_init (G_GNUC_UNUSED MooSyntaxTagClass *klass)
{
}
static void
_moo_syntax_tag_init (MooSyntaxTag *tag)
{
tag->ctx_node = NULL;
tag->match_node = NULL;
tag->rule = NULL;
}
static void
remove_tag (G_GNUC_UNUSED gpointer rule,
GtkTextTag *tag,
GtkTextTagTable *table)
{
gtk_text_tag_table_remove (table, tag);
}
static void
ctx_node_free (CtxNode *node,
GtkTextTagTable *table)
{
if (table)
g_hash_table_foreach (node->match_tags, (GHFunc) remove_tag, table);
g_hash_table_destroy (node->match_tags);
g_hash_table_destroy (node->children);
g_slist_free (node->child_tags);
if (node->context_tag && table)
gtk_text_tag_table_remove (table, node->context_tag);
g_free (node);
}
void
_moo_highlighter_destroy (MooHighlighter *hl,
gboolean delete_tags)
{
GtkTextTagTable *table = NULL;
g_return_if_fail (hl != NULL);
if (delete_tags)
table = gtk_text_buffer_get_tag_table (hl->buffer);
g_slist_foreach (hl->nodes, (GFunc) ctx_node_free, table);
if (hl->lang)
moo_lang_unref (hl->lang);
if (hl->idle)
g_source_remove (hl->idle);
g_free (hl);
}
GtkTextTag *
_moo_text_iter_get_syntax_tag (const GtkTextIter *iter)
{
return (GtkTextTag*) iter_get_syntax_tag (iter);
}
static MooSyntaxTag*
iter_get_syntax_tag (const GtkTextIter *iter)
{
MooSyntaxTag *tag = NULL;
GSList *l;
GSList *list = gtk_text_iter_get_tags (iter);
for (l = list; l != NULL; l = l->next)
{
if (MOO_IS_SYNTAX_TAG (l->data))
{
#ifndef MOO_DEBUG
tag = l->data;
break;
#else
g_assert (tag == NULL);
tag = l->data;
#endif
}
}
g_slist_free (list);
return tag;
}
static void
apply_tag (MooHighlighter *hl,
Line *line,
CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule,
const GtkTextIter *start,
const GtkTextIter *end)
{
GtkTextTag *tag = get_syntax_tag (hl, ctx_node, match_node, rule);
g_assert (!tag || (MOO_IS_SYNTAX_TAG (tag) && MOO_SYNTAX_TAG(tag)->rule == rule));
g_assert (!tag || MOO_SYNTAX_TAG(tag)->match_node != NULL || MOO_SYNTAX_TAG(tag)->ctx_node != hl->root);
if (!gtk_text_iter_compare (start, end))
return;
g_assert (iter_get_syntax_tag (start) == NULL);
if (tag && MOO_SYNTAX_TAG(tag)->has_style)
{
#if 0
g_print ("* applying tag %s-%s-'%s' on line %d\n",
ctx_node ? ctx_node->ctx->name : "<UNKNOWN>",
match_node ? match_node->ctx->name : "<>",
rule ? rule->description : "NULL",
_moo_line_buffer_get_line_index (hl->line_buf, line));
#endif
line->hl_info->tags = g_slist_prepend (line->hl_info->tags, g_object_ref (tag));
_moo_text_buffer_apply_syntax_tag (MOO_TEXT_BUFFER (hl->buffer), tag, start, end);
}
}
MooHighlighter*
_moo_highlighter_new (GtkTextBuffer *buffer,
LineBuffer *line_buf,
MooLang *lang)
{
MooHighlighter *hl;
g_return_val_if_fail (GTK_IS_TEXT_BUFFER (buffer), NULL);
hl = g_new0 (MooHighlighter, 1);
hl->lang = lang ? moo_lang_ref (lang) : NULL;
hl->buffer = buffer;
hl->line_buf = line_buf;
return hl;
}
static GtkTextTag*
moo_syntax_tag_new (CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule)
{
GtkTextTag *tag = g_object_new (MOO_TYPE_SYNTAX_TAG, NULL);
g_assert (ctx_node != NULL);
g_assert (rule != NULL || match_node == NULL);
g_assert (rule == NULL || match_node != NULL);
MOO_SYNTAX_TAG(tag)->ctx_node = ctx_node;
MOO_SYNTAX_TAG(tag)->match_node = match_node;
MOO_SYNTAX_TAG(tag)->rule = rule;
return tag;
}
static GtkTextTag*
create_tag (MooHighlighter *hl,
CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule)
{
GtkTextTag *tag, *cbt = NULL, *ibt = NULL;
GtkTextTagTable *table;
if (!ctx_node->parent && !match_node && !rule)
return NULL;
tag = moo_syntax_tag_new (ctx_node, match_node, rule);
table = gtk_text_buffer_get_tag_table (hl->buffer);
gtk_text_tag_table_add (table, tag);
g_object_unref (tag);
#if 1
/* XXX */
_moo_text_buffer_get_bracket_tags (MOO_TEXT_BUFFER (hl->buffer), &cbt, &ibt);
if (ibt)
gtk_text_tag_set_priority (ibt, gtk_text_tag_table_get_size (table) - 2);
if (cbt)
gtk_text_tag_set_priority (cbt, gtk_text_tag_table_get_size (table) - 1);
#endif
_moo_lang_set_tag_style (tag, ctx_node->ctx, rule, NULL);
ctx_node->child_tags = g_slist_prepend (ctx_node->child_tags, tag);
return tag;
}
static GtkTextTag*
get_syntax_tag (MooHighlighter *hl,
CtxNode *ctx_node,
CtxNode *match_node,
MooRule *rule)
{
GtkTextTag *tag;
g_assert (ctx_node != NULL);
g_assert ((!match_node && !rule) || (match_node && rule));
if (!match_node)
return ctx_node->context_tag;
tag = g_hash_table_lookup (match_node->match_tags, rule);
if (!tag)
{
tag = create_tag (hl, ctx_node, match_node, rule);
g_hash_table_insert (match_node->match_tags, rule, tag);
}
return tag;
}
static CtxNode*
ctx_node_new (MooHighlighter *hl,
CtxNode *parent,
MooContext *context,
gboolean create_text_tag)
{
CtxNode *node = g_new0 (CtxNode, 1);
g_return_val_if_fail (context != NULL, NULL);
node->ctx = context;
node->parent = parent;
node->match_tags = g_hash_table_new (g_direct_hash, g_direct_equal);
node->children = g_hash_table_new (g_direct_hash, g_direct_equal);
if (parent)
node->depth = parent->depth + 1;
else
node->depth = 0;
if (create_text_tag)
node->context_tag = create_tag (hl, node, NULL, NULL);
hl->nodes = g_slist_prepend (hl->nodes, node);
return node;
}
static CtxNode*
get_next_node (MooHighlighter *hl,
CtxNode *node,
MooRule *rule)
{
guint i;
CtxNode *next;
g_assert (rule != NULL);
if ((next = g_hash_table_lookup (node->children, rule)))
return next;
switch (rule->exit.type)
{
case MOO_CONTEXT_STAY:
next = node;
break;
case MOO_CONTEXT_POP:
for (i = 0, next = node; next->parent != NULL && i < rule->exit.num;
++i, next = next->parent);
break;
case MOO_CONTEXT_SWITCH:
next = ctx_node_new (hl, node, rule->exit.ctx, TRUE);
break;
}
g_assert (next != NULL);
g_hash_table_insert (node->children, rule, next);
return next;
}
static CtxNode*
get_line_end_node (MooHighlighter *hl,
CtxNode *node)
{
guint i;
CtxNode *next = NULL;
MooContext *ctx = node->ctx;
if (node->line_end)
return node->line_end;
switch (ctx->line_end.type)
{
case MOO_CONTEXT_STAY:
next = node;
break;
case MOO_CONTEXT_POP:
for (i = 0, next = node; next->parent != NULL && i < ctx->line_end.num;
++i, next = next->parent);
break;
case MOO_CONTEXT_SWITCH:
next = ctx_node_new (hl, node, ctx->line_end.ctx, TRUE);
break;
}
g_assert (next != NULL);
node->line_end = next;
return next;
}
static CtxNode*
get_next_line_node (MooHighlighter *hl,
Line *line)
{
HLInfo *info = line->hl_info;
CtxNode *last_node = NULL;
if (!info->n_segments)
{
last_node = info->start_node;
}
else
{
Segment *last = &info->segments[info->n_segments - 1];
if (last->rule)
{
if (last->rule->include_eol)
return get_next_node (hl, last->ctx_node, last->rule);
else
last_node = get_next_node (hl, last->ctx_node, last->rule);
}
else
{
last_node = last->ctx_node;
}
}
g_assert (last_node != NULL);
return get_line_end_node (hl, last_node);
}
static CtxNode *
hl_compute_line (MooHighlighter *hl,
Line *line,
MatchData *data,
CtxNode *node)
{
MooRule *matched_rule;
MatchResult result;
gboolean nomatch = FALSE;
while (!gtk_text_iter_ends_line (&data->start_iter))
{
if ((matched_rule = _moo_rule_array_match (node->ctx->rules, data, &result)))
{
CtxNode *match_node, *next_node;
if (result.match_offset < 0)
result.match_offset = g_utf8_pointer_to_offset (data->start, result.match_start);
if (result.match_len < 0)
result.match_len = g_utf8_pointer_to_offset (result.match_start, result.match_end);
next_node = get_next_node (hl, node, matched_rule);
if (next_node == node && !result.match_offset && !result.match_len)
{
g_critical ("%s: zero-length match returned to the same node, bailing out", G_STRLOC);
nomatch = TRUE;
break;
}
if (result.match_offset)
_moo_line_add_segment (line, result.match_offset, node, NULL, NULL);
if (matched_rule->flags & MOO_RULE_INCLUDE_INTO_NEXT)
match_node = next_node;
else
match_node = node;
_moo_line_add_segment (line, result.match_len, match_node, node, matched_rule);
_moo_match_data_set_start (data, NULL, result.match_end,
data->start_offset + result.match_offset + result.match_len);
node = next_node;
}
else
{
nomatch = TRUE;
break;
}
}
if (nomatch)
_moo_line_add_segment (line, -1, node, NULL, NULL);
return get_next_line_node (hl, line);
}
static gboolean
hl_compute_range (MooHighlighter *hl,
Interval *lines,
int time)
{
GtkTextIter iter;
CtxNode *node = NULL;
int line_no;
GTimer *timer = NULL;
double secs = ((double) time) / 1000;
gboolean done = FALSE;
g_assert (!hl->line_buf->invalid.empty);
g_assert (lines->first == hl->line_buf->invalid.first);
if (time > 0)
timer = g_timer_new ();
if (lines->first == 0)
{
if (!hl->root)
hl->root = ctx_node_new (hl, NULL,
_moo_lang_get_default_context (hl->lang),
FALSE);
node = hl->root;
}
else
{
Line *prev = _moo_line_buffer_get_line (hl->line_buf, lines->first - 1);
node = get_next_line_node (hl, prev);
}
g_return_val_if_fail (node != NULL, TRUE);
gtk_text_buffer_get_iter_at_line (hl->buffer, &iter, lines->first);
#if 0
g_print ("hl_compute_range: %d to %d\n", lines->first, lines->last);
g_print ("hl_compute_range: invalid %d to %d\n",
hl->line_buf->invalid.first,
hl->line_buf->invalid.last);
#endif
for (line_no = lines->first; line_no <= lines->last; ++line_no)
{
Line *line = _moo_line_buffer_get_line (hl->line_buf, line_no);
MatchData match_data;
g_assert (line_no == gtk_text_iter_get_line (&iter));
line->hl_info->start_node = node;
line->hl_info->tags_applied = FALSE;
_moo_line_erase_segments (line);
/* TODO: there is no need to recompute line if its start context matches
context implied by the previous line */
_moo_match_data_init (&match_data, line_no, &iter, NULL);
node = hl_compute_line (hl, line, &match_data, node);
_moo_match_data_destroy (&match_data);
#if 0
{
guint i;
g_print ("line %d, %d segments\n", line_no, line->hl_info->n_segments);
for (i = 0; i < line->hl_info->n_segments; ++i)
{
Segment *s = &line->hl_info->segments[i];
g_print ("segment %d: %d chars, context '%s'", i, s->len, s->ctx_node->ctx->name);
if (s->match_node)
g_print (", match context '%s'", s->match_node->ctx->name);
if (s->rule)
g_print (", rule '%s'", s->rule->description);
g_print ("\n");
}
}
#endif
if (!gtk_text_iter_forward_line (&iter))
{
g_assert (line_no >= lines->last - 1); /* if last line is empty,
then this last line 'does not
really exist'*/
g_assert (line_no >= hl->line_buf->invalid.last - 1);
done = TRUE;
break;
}
if (line_no == hl->line_buf->invalid.last)
{
Line *next = _moo_line_buffer_get_line (hl->line_buf, line_no + 1);
if (node == next->hl_info->start_node)
{
done = TRUE;
break;
}
else
{
hl->line_buf->invalid.last++;
}
}
if (timer && (g_timer_elapsed (timer, NULL) > secs))
break;
}
if (!done)
{
hl->line_buf->invalid.first = line_no;
if (line_no <= lines->last)
hl->line_buf->invalid.first++;
_moo_line_buffer_clamp_invalid (hl->line_buf);
}
else
{
BUF_SET_CLEAN (hl->line_buf);
}
if (timer)
g_timer_destroy (timer);
_moo_text_buffer_highlighting_changed (MOO_TEXT_BUFFER (hl->buffer),
lines->first, line_no);
#if 0
g_print ("changed %d to %d\n", lines->first, line_no);
#endif
return done;
}
static gboolean
moo_highlighter_compute_timed (MooHighlighter *hl,
int first_line,
int last_line,
int time)
{
Interval to_highlight;
if (!hl->lang || !hl->buffer)
return TRUE;
if (BUF_CLEAN (hl->line_buf))
return TRUE;
if (last_line < 0)
last_line = _moo_text_btree_size (hl->line_buf->tree) - 1;
g_assert (first_line >= 0);
g_assert (last_line >= first_line);
to_highlight.first = first_line;
to_highlight.last = last_line;
if (hl->line_buf->invalid.first > to_highlight.last)
return TRUE;
to_highlight.first = hl->line_buf->invalid.first;
return hl_compute_range (hl, &to_highlight, time);
}
static gboolean
compute_in_idle (MooHighlighter *hl)
{
hl->idle = 0;
if (!BUF_CLEAN (hl->line_buf))
moo_highlighter_compute_timed (hl, 0, -1,
IDLE_HIGHLIGHT_TIME);
if (!BUF_CLEAN (hl->line_buf))
hl->idle = g_timeout_add_full (IDLE_HIGHLIGHT_PRIORITY,
IDLE_HIGHLIGHT_TIME,
(GSourceFunc) compute_in_idle,
hl, NULL);
return FALSE;
}
void
_moo_highlighter_queue_compute (MooHighlighter *hl)
{
if (!hl->lang || !hl->buffer || BUF_CLEAN (hl->line_buf))
return;
if (!hl->idle)
hl->idle = g_idle_add_full (IDLE_HIGHLIGHT_PRIORITY,
(GSourceFunc) compute_in_idle,
hl, NULL);
}
void
_moo_highlighter_apply_tags (MooHighlighter *hl,
int first_line,
int last_line)
{
int line_no;
int total;
int first_changed, last_changed;
if (!hl->lang || !hl->buffer)
return;
total = _moo_text_btree_size (hl->line_buf->tree);
if (last_line < 0)
last_line = total - 1;
first_line = CLAMP (first_line, 0, total - 1);
last_line = CLAMP (last_line, first_line, total - 1);
g_assert (first_line >= 0);
g_assert (last_line >= first_line);
if (!moo_highlighter_compute_timed (hl, first_line, last_line,
COMPUTE_NOW_TIME))
_moo_highlighter_queue_compute (hl);
first_changed = last_changed = -1;
for (line_no = first_line; line_no <= last_line; ++line_no)
{
Line *line = _moo_line_buffer_get_line (hl->line_buf, line_no);
HLInfo *info = line->hl_info;
guint i;
GtkTextIter t_start, t_end;
GSList *tags;
if (!hl->line_buf->invalid.empty && line_no >= hl->line_buf->invalid.first)
break;
if (line->hl_info->tags_applied)
continue;
if (first_changed < 0)
first_changed = line_no;
last_changed = line_no;
line->hl_info->tags_applied = TRUE;
tags = line->hl_info->tags;
line->hl_info->tags = NULL;
gtk_text_buffer_get_iter_at_line (hl->buffer, &t_start, line_no);
if (gtk_text_iter_ends_line (&t_start))
{
g_slist_foreach (tags, (GFunc) g_object_unref, NULL);
g_slist_free (tags);
continue;
}
t_end = t_start;
gtk_text_iter_forward_to_line_end (&t_end);
while (tags)
{
gtk_text_buffer_remove_tag (hl->buffer, tags->data, &t_start, &t_end);
g_object_unref (tags->data);
tags = g_slist_delete_link (tags, tags);
}
t_end = t_start;
for (i = 0; i < info->n_segments; ++i)
{
if (!info->segments[i].len)
continue;
if (gtk_text_iter_ends_line (&t_start))
{
g_assert (info->segments[i].len < 0);
break;
}
if (info->segments[i].len < 0)
gtk_text_iter_forward_to_line_end (&t_end);
else
gtk_text_iter_forward_chars (&t_end, info->segments[i].len);
apply_tag (hl, line, info->segments[i].ctx_node,
info->segments[i].match_node,
info->segments[i].rule,
&t_start, &t_end);
t_start = t_end;
}
}
if (first_changed >= 0)
_moo_text_buffer_tags_changed (MOO_TEXT_BUFFER (hl->buffer),
first_changed, last_changed);
}
static void
tag_set_scheme (G_GNUC_UNUSED gpointer whatever,
MooSyntaxTag *tag,
MooTextStyleScheme *scheme)
{
if (tag)
{
_moo_lang_erase_tag_style (GTK_TEXT_TAG (tag));
_moo_lang_set_tag_style (GTK_TEXT_TAG (tag),
tag->ctx_node->ctx,
tag->rule, scheme);
}
}
static void
ctx_node_set_scheme (CtxNode *node,
MooTextStyleScheme *scheme)
{
g_hash_table_foreach (node->match_tags, (GHFunc) tag_set_scheme, scheme);
tag_set_scheme (NULL, MOO_SYNTAX_TAG (node->context_tag), scheme);
}
void
_moo_highlighter_apply_scheme (MooHighlighter *hl,
MooTextStyleScheme *scheme)
{
g_return_if_fail (hl != NULL && scheme != NULL);
g_slist_foreach (hl->nodes, (GFunc) ctx_node_set_scheme, scheme);
}