707 lines
19 KiB
C
707 lines
19 KiB
C
/*
|
|
* moofold.c
|
|
*
|
|
* Copyright (C) 2004-2008 by Yevgen Muntyan <muntyan@tamu.edu>
|
|
* Copyright (C) 2008 by Daniel Poelzleithner <mooedit@poelzi.org>
|
|
*
|
|
* This file is part of medit. medit is free software; you can
|
|
* redistribute it and/or modify it under the terms of the
|
|
* GNU Lesser General Public License as published by the
|
|
* Free Software Foundation; either version 2.1 of the License,
|
|
* or (at your option) any later version.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#define MOOEDIT_COMPILATION
|
|
#include "mooedit/mootext-private.h"
|
|
#include "marshals.h"
|
|
|
|
|
|
#ifdef MOO_DEBUG_ENABLED
|
|
#define WANT_CHECKS 1
|
|
#else
|
|
#define WANT_CHECKS 0
|
|
#endif
|
|
|
|
static void moo_fold_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void moo_fold_finalize (GObject *object);
|
|
static void moo_fold_free_recursively (MooFold *fold);
|
|
|
|
static int _moo_fold_get_end (MooFold *fold);
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_PARENT,
|
|
PROP_NEXT,
|
|
PROP_PREV,
|
|
PROP_CHILDREN,
|
|
PROP_N_CHILDREN,
|
|
PROP_MARK_START,
|
|
PROP_MARK_END,
|
|
PROP_COLLAPSED
|
|
};
|
|
|
|
/* MOO_TYPE_FOLD */
|
|
G_DEFINE_TYPE (MooFold, moo_fold, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
moo_fold_class_init (MooFoldClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->finalize = moo_fold_finalize;
|
|
|
|
gobject_class->get_property = moo_fold_get_property;
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PARENT,
|
|
g_param_spec_object ("parent",
|
|
"parent",
|
|
"parent",
|
|
MOO_TYPE_FOLD,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_NEXT,
|
|
g_param_spec_object ("next",
|
|
"next",
|
|
"next",
|
|
MOO_TYPE_FOLD,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PREV,
|
|
g_param_spec_object ("previous",
|
|
"previous",
|
|
"previous",
|
|
MOO_TYPE_FOLD,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_MARK_START,
|
|
g_param_spec_object ("mark_start",
|
|
"Start Marker",
|
|
"Marker of the start Marker",
|
|
MOO_TYPE_LINE_MARK,
|
|
G_PARAM_READABLE));
|
|
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_MARK_END,
|
|
g_param_spec_object ("mark_end",
|
|
"End Marker",
|
|
"Marker of the end Marker",
|
|
MOO_TYPE_LINE_MARK,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_CHILDREN,
|
|
g_param_spec_object ("children",
|
|
"Children",
|
|
"Children ordered by Line",
|
|
MOO_TYPE_FOLD,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_N_CHILDREN,
|
|
g_param_spec_int ("n_children",
|
|
"Children Number",
|
|
"Number of children",
|
|
0, G_MAXINT, 0,
|
|
G_PARAM_READABLE));
|
|
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_COLLAPSED,
|
|
g_param_spec_boolean ("collapsed",
|
|
"Fold Collapsed",
|
|
"True if the fold is collapsed",
|
|
FALSE,
|
|
G_PARAM_READABLE));
|
|
}
|
|
|
|
static void
|
|
moo_fold_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
MooFold *fold = MOO_FOLD (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_PARENT:
|
|
g_value_set_object (value, fold->parent);
|
|
break;
|
|
|
|
case PROP_NEXT:
|
|
g_value_set_object (value, fold->next);
|
|
break;
|
|
|
|
case PROP_PREV:
|
|
g_value_set_object (value, fold->prev);
|
|
break;
|
|
|
|
case PROP_MARK_START:
|
|
g_value_set_object (value, fold->start);
|
|
break;
|
|
|
|
case PROP_CHILDREN:
|
|
g_value_set_object (value, fold->children);
|
|
break;
|
|
|
|
case PROP_N_CHILDREN:
|
|
g_value_set_int (value, fold->n_children);
|
|
break;
|
|
|
|
case PROP_MARK_END:
|
|
g_value_set_object (value, fold->end);
|
|
break;
|
|
|
|
case PROP_COLLAPSED:
|
|
g_value_set_boolean (value, fold->collapsed);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
moo_fold_init (MooFold *fold)
|
|
{
|
|
fold->deleted = TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_fold_finalize (GObject *object)
|
|
{
|
|
MooFold *fold = MOO_FOLD (object);
|
|
|
|
if (!fold->deleted)
|
|
g_critical ("%s: oops, crash pending...", G_STRLOC);
|
|
|
|
G_OBJECT_CLASS (moo_fold_parent_class)->finalize (object);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_fold_free_recursively (MooFold *fold)
|
|
{
|
|
MooFold *child;
|
|
|
|
g_return_if_fail (MOO_IS_FOLD (fold));
|
|
|
|
fold->deleted = TRUE;
|
|
|
|
child = fold->children;
|
|
fold->children = NULL;
|
|
|
|
for ( ; child != NULL; child = child->next)
|
|
moo_fold_free_recursively (child);
|
|
|
|
if (fold->start)
|
|
g_object_unref (fold->start);
|
|
if (fold->end)
|
|
g_object_unref (fold->end);
|
|
g_object_unref (fold);
|
|
}
|
|
|
|
|
|
int
|
|
_moo_fold_get_start (MooFold *fold)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLD (fold), -1);
|
|
g_return_val_if_fail (!_moo_fold_is_deleted (fold), -1);
|
|
return moo_line_mark_get_line (fold->start);
|
|
}
|
|
|
|
|
|
static int
|
|
_moo_fold_get_end (MooFold *fold)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLD (fold), -1);
|
|
g_return_val_if_fail (!_moo_fold_is_deleted (fold), -1);
|
|
return moo_line_mark_get_line (fold->end);
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_fold_is_deleted (MooFold *fold)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_FOLD (fold), TRUE);
|
|
return fold->deleted != 0;
|
|
}
|
|
|
|
|
|
#if WANT_CHECKS
|
|
|
|
static void
|
|
CHECK_LIST_MEMBER__ (MooFold *list,
|
|
MooFold *fold)
|
|
{
|
|
while (list)
|
|
{
|
|
if (list == fold)
|
|
return;
|
|
list = list->next;
|
|
}
|
|
|
|
g_assert_not_reached ();
|
|
}
|
|
|
|
|
|
static void
|
|
CHECK_FOLD (MooFoldTree *tree,
|
|
MooFold *fold)
|
|
{
|
|
g_assert (tree != NULL);
|
|
g_assert (MOO_IS_FOLD (fold));
|
|
g_assert (!fold->deleted);
|
|
|
|
while (fold->parent)
|
|
{
|
|
CHECK_LIST_MEMBER__ (fold->parent->children, fold);
|
|
fold = fold->parent;
|
|
}
|
|
|
|
CHECK_LIST_MEMBER__ (tree->folds, fold);
|
|
}
|
|
|
|
#else /* WANT_CHECKS */
|
|
#define CHECK_FOLD(tree,fold)
|
|
#endif
|
|
|
|
|
|
MooFoldTree *
|
|
_moo_fold_tree_new (MooTextBuffer *buffer)
|
|
{
|
|
MooFoldTree *tree = g_new0 (MooFoldTree, 1);
|
|
|
|
tree->buffer = buffer;
|
|
tree->consistent = TRUE;
|
|
|
|
return tree;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_fold_tree_free (MooFoldTree *tree)
|
|
{
|
|
MooFold *child;
|
|
|
|
g_return_if_fail (tree != NULL);
|
|
|
|
for (child = tree->folds; child != NULL; child = child->next)
|
|
moo_fold_free_recursively (child);
|
|
|
|
g_free (tree);
|
|
}
|
|
|
|
|
|
inline static int
|
|
get_line_count (MooFoldTree *tree)
|
|
{
|
|
return gtk_text_buffer_get_line_count (GTK_TEXT_BUFFER (tree->buffer));
|
|
}
|
|
|
|
|
|
static MooLineMark *
|
|
fold_mark_new (MooFold *fold)
|
|
{
|
|
MooLineMark *mark = g_object_new (MOO_TYPE_LINE_MARK, NULL);
|
|
_moo_line_mark_set_fold (mark, fold);
|
|
return mark;
|
|
}
|
|
|
|
|
|
static MooFold *
|
|
insert_fold (MooFoldTree *tree,
|
|
MooFold *parent,
|
|
int first_line,
|
|
int last_line)
|
|
{
|
|
MooFold *fold, *last, *new_fold;
|
|
|
|
g_assert (!parent || MOO_IS_FOLD (parent));
|
|
g_assert (!parent || _moo_fold_get_start (parent) < first_line);
|
|
g_assert (!parent || _moo_fold_get_end (parent) >= last_line);
|
|
|
|
last = NULL;
|
|
|
|
for (fold = (parent ? parent->children : tree->folds); fold != NULL; fold = fold->next)
|
|
{
|
|
int start, end;
|
|
|
|
start = _moo_fold_get_start (fold);
|
|
end = _moo_fold_get_end (fold);
|
|
|
|
if (!fold->next)
|
|
last = fold;
|
|
|
|
if (end < first_line)
|
|
continue;
|
|
|
|
if (start > last_line)
|
|
break;
|
|
|
|
if (start == first_line || end == first_line || start == last_line)
|
|
return NULL;
|
|
|
|
if (start < first_line)
|
|
{
|
|
if (end < last_line)
|
|
return NULL;
|
|
|
|
/* new fold is inserted into the old one */
|
|
return insert_fold (tree, fold, first_line, last_line);
|
|
}
|
|
else /* start > first_line */
|
|
{
|
|
if (end > last_line)
|
|
return NULL;
|
|
|
|
/* old fold is inserted into the new one */
|
|
|
|
new_fold = g_object_new (MOO_TYPE_FOLD, NULL);
|
|
new_fold->deleted = FALSE;
|
|
|
|
if (parent)
|
|
{
|
|
if (!parent->children || fold == parent->children)
|
|
parent->children = new_fold;
|
|
}
|
|
else
|
|
{
|
|
if (!tree->folds || fold == tree->folds)
|
|
tree->folds = new_fold;
|
|
}
|
|
|
|
new_fold->parent = parent;
|
|
fold->parent = new_fold;
|
|
new_fold->children = fold;
|
|
new_fold->n_children = 1;
|
|
|
|
if (fold->next)
|
|
{
|
|
fold->next->prev = new_fold;
|
|
new_fold->next = fold->next;
|
|
fold->next = NULL;
|
|
}
|
|
|
|
if (fold->prev)
|
|
{
|
|
fold->prev->next = new_fold;
|
|
new_fold->prev = fold->prev;
|
|
fold->prev = NULL;
|
|
}
|
|
|
|
new_fold->start = fold_mark_new (new_fold);
|
|
new_fold->end = fold_mark_new (new_fold);
|
|
|
|
moo_text_buffer_add_line_mark (tree->buffer, new_fold->start, first_line);
|
|
moo_text_buffer_add_line_mark (tree->buffer, new_fold->end, last_line);
|
|
|
|
CHECK_FOLD (tree, new_fold);
|
|
return new_fold;
|
|
}
|
|
}
|
|
|
|
/* insert new fold before <fold> or in the end if <fold> is NULL */
|
|
|
|
new_fold = g_object_new (MOO_TYPE_FOLD, NULL);
|
|
new_fold->deleted = FALSE;
|
|
|
|
new_fold->parent = parent;
|
|
new_fold->next = fold;
|
|
|
|
if (parent)
|
|
{
|
|
if (fold == parent->children)
|
|
parent->children = new_fold;
|
|
parent->n_children++;
|
|
}
|
|
else
|
|
{
|
|
if (fold == tree->folds)
|
|
tree->folds = new_fold;
|
|
tree->n_folds++;
|
|
}
|
|
|
|
if (fold)
|
|
{
|
|
if (fold->prev)
|
|
new_fold->prev = fold->prev;
|
|
fold->prev = new_fold;
|
|
}
|
|
else if (last)
|
|
{
|
|
g_assert (last->next == NULL);
|
|
last->next = new_fold;
|
|
new_fold->prev = last;
|
|
}
|
|
|
|
new_fold->start = fold_mark_new (new_fold);
|
|
new_fold->end = fold_mark_new (new_fold);
|
|
|
|
moo_text_buffer_add_line_mark (tree->buffer, new_fold->start, first_line);
|
|
moo_text_buffer_add_line_mark (tree->buffer, new_fold->end, last_line);
|
|
|
|
CHECK_FOLD (tree, new_fold);
|
|
return new_fold;
|
|
}
|
|
|
|
|
|
|
|
MooFold *
|
|
_moo_fold_tree_add (MooFoldTree *tree,
|
|
int first_line,
|
|
int last_line)
|
|
{
|
|
g_assert (tree != NULL);
|
|
g_assert (first_line >= 0 && first_line < get_line_count (tree));
|
|
g_assert (last_line >= 0 && last_line < get_line_count (tree));
|
|
g_assert (last_line > first_line);
|
|
return insert_fold (tree, NULL, first_line, last_line);
|
|
}
|
|
|
|
|
|
static void
|
|
fold_free (MooFold *fold)
|
|
{
|
|
fold->deleted = TRUE;
|
|
|
|
if (fold->start)
|
|
{
|
|
_moo_line_mark_set_fold (fold->start, NULL);
|
|
|
|
if (!moo_line_mark_get_deleted (fold->start))
|
|
moo_text_buffer_delete_line_mark (moo_line_mark_get_buffer (fold->start),
|
|
fold->start);
|
|
|
|
g_object_unref (fold->start);
|
|
fold->start = NULL;
|
|
}
|
|
|
|
if (fold->end)
|
|
{
|
|
_moo_line_mark_set_fold (fold->end, NULL);
|
|
|
|
if (!moo_line_mark_get_deleted (fold->end))
|
|
moo_text_buffer_delete_line_mark (moo_line_mark_get_buffer (fold->end),
|
|
fold->end);
|
|
|
|
g_object_unref (fold->end);
|
|
fold->end = NULL;
|
|
}
|
|
|
|
g_object_unref (fold);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_fold_tree_remove (MooFoldTree *tree,
|
|
MooFold *fold)
|
|
{
|
|
guint n_children;
|
|
MooFold *children, *parent, *last;
|
|
|
|
CHECK_FOLD (tree, fold);
|
|
|
|
_moo_fold_tree_expand (tree, fold);
|
|
|
|
n_children = fold->n_children;
|
|
last = children = fold->children;
|
|
parent = fold->parent;
|
|
|
|
if (children)
|
|
{
|
|
while (TRUE)
|
|
{
|
|
g_assert (last->parent == fold);
|
|
last->parent = fold->parent;
|
|
if (!last->next)
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_assert ((last && children) || (!last && !children));
|
|
|
|
if (parent)
|
|
{
|
|
if (fold == parent->children)
|
|
parent->children = children ? children : parent->children->next;
|
|
parent->n_children = parent->n_children + n_children - 1;
|
|
}
|
|
else
|
|
{
|
|
if (fold == tree->folds)
|
|
tree->folds = children ? children : tree->folds->next;
|
|
tree->n_folds = tree->n_folds + n_children - 1;
|
|
}
|
|
|
|
if (last)
|
|
last->next = fold->next;
|
|
if (fold->next)
|
|
fold->next->prev = last ? last : fold->prev;
|
|
if (children)
|
|
children->prev = fold->prev;
|
|
if (fold->prev)
|
|
fold->prev->next = children ? children : fold->next;
|
|
|
|
fold_free (fold);
|
|
}
|
|
|
|
|
|
static void
|
|
expand_check_visible (MooFoldTree *tree,
|
|
MooFold *fold)
|
|
{
|
|
GtkTextIter start, end;
|
|
GtkTextBuffer *buffer;
|
|
MooFold *child;
|
|
|
|
CHECK_FOLD (tree, fold);
|
|
|
|
buffer = GTK_TEXT_BUFFER (tree->buffer);
|
|
gtk_text_buffer_get_iter_at_line (buffer, &start,
|
|
_moo_fold_get_start (fold));
|
|
end = start;
|
|
gtk_text_iter_forward_line (&end);
|
|
gtk_text_buffer_remove_tag_by_name (buffer, MOO_FOLD_TAG, &start, &end);
|
|
|
|
if (fold->collapsed)
|
|
return;
|
|
|
|
for (child = fold->children; child != NULL; child = child->next)
|
|
{
|
|
int start_line, end_line;
|
|
|
|
if (child->prev)
|
|
start_line = _moo_fold_get_end (child->prev) + 1;
|
|
else
|
|
start_line = _moo_fold_get_start (fold) + 1;
|
|
|
|
end_line = _moo_fold_get_start (child);
|
|
|
|
gtk_text_buffer_get_iter_at_line (buffer, &start, start_line);
|
|
gtk_text_buffer_get_iter_at_line (buffer, &end, end_line);
|
|
gtk_text_iter_forward_line (&end);
|
|
gtk_text_buffer_remove_tag_by_name (buffer, MOO_FOLD_TAG, &start, &end);
|
|
|
|
expand_check_visible (tree, child);
|
|
|
|
if (!child->next)
|
|
break;
|
|
}
|
|
|
|
if (child)
|
|
{
|
|
if (_moo_fold_get_end (child) < _moo_fold_get_end (fold))
|
|
{
|
|
gtk_text_buffer_get_iter_at_line (buffer, &start,
|
|
_moo_fold_get_end (child) + 1);
|
|
gtk_text_buffer_get_iter_at_line (buffer, &end,
|
|
_moo_fold_get_end (fold));
|
|
gtk_text_iter_forward_line (&end);
|
|
gtk_text_buffer_remove_tag_by_name (buffer, MOO_FOLD_TAG, &start, &end);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
start = end;
|
|
gtk_text_buffer_get_iter_at_line (buffer, &end,
|
|
_moo_fold_get_end (fold));
|
|
gtk_text_iter_forward_line (&end);
|
|
gtk_text_buffer_remove_tag_by_name (buffer, MOO_FOLD_TAG, &start, &end);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
_moo_fold_tree_expand (MooFoldTree *tree,
|
|
MooFold *fold)
|
|
{
|
|
CHECK_FOLD (tree, fold);
|
|
|
|
if (!fold->collapsed)
|
|
return;
|
|
|
|
fold->collapsed = FALSE;
|
|
expand_check_visible (tree, fold);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_fold_tree_collapse (MooFoldTree *tree,
|
|
MooFold *fold)
|
|
{
|
|
GtkTextIter start, end;
|
|
GtkTextBuffer *buffer;
|
|
|
|
CHECK_FOLD (tree, fold);
|
|
|
|
if (fold->collapsed)
|
|
return;
|
|
|
|
fold->collapsed = TRUE;
|
|
|
|
buffer = GTK_TEXT_BUFFER (tree->buffer);
|
|
gtk_text_buffer_get_iter_at_line (buffer, &start,
|
|
_moo_fold_get_start (fold) + 1);
|
|
gtk_text_buffer_get_iter_at_line (buffer, &end,
|
|
_moo_fold_get_end (fold));
|
|
gtk_text_iter_forward_line (&end);
|
|
gtk_text_buffer_apply_tag_by_name (buffer, MOO_FOLD_TAG, &start, &end);
|
|
}
|
|
|
|
|
|
static GSList *
|
|
get_folds_in_range (MooFoldTree *tree,
|
|
MooFold *parent,
|
|
int first_line,
|
|
int last_line,
|
|
GSList *list)
|
|
{
|
|
MooFold *child;
|
|
|
|
if (parent && first_line <= _moo_fold_get_start (parent) && last_line >= _moo_fold_get_start (parent))
|
|
list = g_slist_prepend (list, parent);
|
|
|
|
for (child = parent ? parent->children : tree->folds; child != NULL; child = child->next)
|
|
{
|
|
if (last_line < _moo_fold_get_start (child))
|
|
break;
|
|
if (first_line >= _moo_fold_get_end (child))
|
|
continue;
|
|
list = get_folds_in_range (tree, child, first_line, last_line, list);
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
|
|
GSList *
|
|
_moo_fold_tree_get (MooFoldTree *tree,
|
|
int first_line,
|
|
int last_line)
|
|
{
|
|
GSList *list;
|
|
|
|
g_assert (tree != NULL);
|
|
g_assert (first_line >= 0 && first_line < get_line_count (tree));
|
|
g_assert (last_line >= 0 && last_line < get_line_count (tree));
|
|
g_assert (last_line >= first_line);
|
|
|
|
list = get_folds_in_range (tree, NULL, first_line, last_line, NULL);
|
|
return g_slist_reverse (list);
|
|
}
|