/* * moofold.c * * Copyright (C) 2004-2010 by Yevgen Muntyan * Copyright (C) 2008 by Daniel Poelzleithner * * 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 . */ #define MOOEDIT_COMPILATION #include "mooedit/mootext-private.h" #include "marshals.h" #ifdef MOO_DEBUG #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 ("oops, crash pending..."); 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 = MOO_LINE_MARK (g_object_new (MOO_TYPE_LINE_MARK, (const char*) 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 = MOO_FOLD (g_object_new (MOO_TYPE_FOLD, (const char*) 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 or in the end if is NULL */ new_fold = MOO_FOLD (g_object_new (MOO_TYPE_FOLD, (const char*) 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); } #define MOO_FOLD_FOREACH_BEGIN(start, iter) \ do { \ MooFold *iter = (start); \ while (iter) \ { #define MOO_FOLD_FOREACH_END(iter) \ if (iter->children) \ iter = iter->children; \ else if (iter->next) \ iter = iter->next; \ else \ { \ while (iter->parent && !iter->parent->next) \ iter = iter->parent; \ iter = iter && iter->parent ? \ iter->parent->next : NULL; \ } \ } \ } while (0) gboolean _moo_fold_tree_expand_all (MooFoldTree *tree) { gboolean ret = FALSE; MooFold *fold_first; fold_first = NULL; MOO_FOLD_FOREACH_BEGIN(tree->folds, fold) { if (!fold->deleted && fold->collapsed) { if (!fold_first) fold_first = fold; fold->collapsed = FALSE; } } MOO_FOLD_FOREACH_END(fold); if (fold_first) { GtkTextIter start, end; GtkTextBuffer *buffer = GTK_TEXT_BUFFER (tree->buffer); int line_first; line_first = _moo_fold_get_start (fold_first); gtk_text_buffer_get_iter_at_line (buffer, &start, line_first); gtk_text_buffer_get_end_iter (buffer, &end); gtk_text_buffer_remove_tag_by_name (buffer, MOO_FOLD_TAG, &start, &end); ret = TRUE; } return ret; } 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); } gboolean _moo_fold_tree_collapse_all (MooFoldTree *tree) { MooFold *fold; gboolean ret = FALSE; for (fold = tree->folds; fold != NULL; fold = fold->next) { if (!fold->collapsed) ret = TRUE; _moo_fold_tree_collapse (tree, fold); } return ret; } gboolean _moo_fold_tree_toggle (MooFoldTree *tree) { gboolean seen_expanded = FALSE; gboolean seen_collapsed = FALSE; MOO_FOLD_FOREACH_BEGIN(tree->folds, fold) { if (!fold->deleted) { if (fold->collapsed) seen_collapsed = TRUE; else seen_expanded = TRUE; } if (seen_collapsed && seen_expanded) break; } MOO_FOLD_FOREACH_END(fold); if (seen_collapsed) { _moo_fold_tree_expand_all (tree); return TRUE; } else if (seen_expanded) { _moo_fold_tree_collapse_all (tree); return TRUE; } return FALSE; } 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); }