Improve symbols_get_current_function() a lot and make it more flexible

Finding the current function now better handles the case the current
line is after a function but outside its scope, and many other issues
the scope reporting had.
This commit is contained in:
Colomban Wendling 2012-09-17 22:41:24 +02:00
parent 9d2dab8fcf
commit 491a45f614

View File

@ -1959,20 +1959,22 @@ static gint get_function_fold_number(GeanyDocument *doc)
} }
/* Should be used only with symbols_get_current_function. */ /* Should be used only with get_current_tag_cached.
static gboolean current_function_changed(GeanyDocument *doc, gint cur_line, gint fold_level) * tag_types caching might trigger recomputation too often but this isn't used differently often
* enough to be an issue for now */
static gboolean current_tag_changed(GeanyDocument *doc, gint cur_line, gint fold_level, guint tag_types)
{ {
static gint old_line = -2; static gint old_line = -2;
static GeanyDocument *old_doc = NULL; static GeanyDocument *old_doc = NULL;
static gint old_fold_num = -1; static gint old_fold_num = -1;
static guint old_tag_types = 0;
const gint fold_num = fold_level & SC_FOLDLEVELNUMBERMASK; const gint fold_num = fold_level & SC_FOLDLEVELNUMBERMASK;
gboolean ret; gboolean ret;
/* check if the cached line and file index have changed since last time: */ /* check if the cached line and file index have changed since last time: */
if (doc == NULL || doc != old_doc) if (doc == NULL || doc != old_doc || old_tag_types != tag_types)
ret = TRUE; ret = TRUE;
else else if (cur_line == old_line)
if (cur_line == old_line)
ret = FALSE; ret = FALSE;
else else
{ {
@ -1989,6 +1991,7 @@ static gboolean current_function_changed(GeanyDocument *doc, gint cur_line, gint
old_line = cur_line; old_line = cur_line;
old_doc = doc; old_doc = doc;
old_fold_num = fold_num; old_fold_num = fold_num;
old_tag_types = tag_types;
return ret; return ret;
} }
@ -2069,83 +2072,79 @@ static gchar *parse_cpp_function_at_line(ScintillaObject *sci, gint tag_line)
} }
/* Sets *tagname to point at the current function or tag name. static gint get_fold_header_after(ScintillaObject *sci, gint line)
* If doc is NULL, reset the cached current tag data to ensure it will be reparsed on the next
* call to this function.
* Returns: line number of the current tag, or -1 if unknown. */
gint symbols_get_current_function(GeanyDocument *doc, const gchar **tagname)
{ {
static gint tag_line = -1; gint line_count = sci_get_line_count(sci);
static gchar *cur_tag = NULL;
gint line;
gint fold_level;
TMWorkObject *tm_file;
if (doc == NULL) /* reset current function */ for (; line < line_count; line++)
{ {
current_function_changed(NULL, -1, -1); if (sci_get_fold_level(sci, line) & SC_FOLDLEVELHEADERFLAG)
g_free(cur_tag); return line;
cur_tag = g_strdup(_("unknown"));
if (tagname != NULL)
*tagname = cur_tag;
tag_line = -1;
return tag_line;
} }
return -1;
}
static gint get_current_tag_name(GeanyDocument *doc, gchar **tagname, guint tag_types)
{
gint line;
gint parent;
line = sci_get_current_line(doc->editor->sci); line = sci_get_current_line(doc->editor->sci);
fold_level = sci_get_fold_level(doc->editor->sci, line); parent = sci_get_fold_parent(doc->editor->sci, line);
/* check if the cached line and file index have changed since last time: */ /* if we're inside a fold level and we have up-to-date tags, get the function from TM */
if (! current_function_changed(doc, line, fold_level)) if (parent >= 0 && doc->tm_file != NULL && doc->tm_file->tags_array != NULL &&
{
/* we can assume same current function as before */
*tagname = cur_tag;
return tag_line;
}
g_free(cur_tag); /* free the old tag, it will be replaced. */
/* if line is at base fold level, we're not in a function */
if ((fold_level & SC_FOLDLEVELNUMBERMASK) == SC_FOLDLEVELBASE)
{
cur_tag = g_strdup(_("unknown"));
*tagname = cur_tag;
tag_line = -1;
return tag_line;
}
tm_file = doc->tm_file;
/* if the tags are up-to-date, get the previous function name from TM */
if (tm_file != NULL && tm_file->tags_array != NULL &&
(! doc->changed || editor_prefs.autocompletion_update_freq > 0)) (! doc->changed || editor_prefs.autocompletion_update_freq > 0))
{ {
const TMTag *tag = (const TMTag*) tm_get_current_function(tm_file->tags_array, line + 1); const TMTag *tag = tm_get_current_tag(doc->tm_file->tags_array, parent + 1, tag_types);
if (tag != NULL) if (tag)
{ {
gchar *tmp; gint tag_line = tag->atts.entry.line - 1;
tmp = tag->atts.entry.scope; gint last_child = line + 1;
cur_tag = tmp ? g_strconcat(tmp, "::", tag->name, NULL) : g_strdup(tag->name);
*tagname = cur_tag; /* if it may be a false positive because we're inside a fold level not inside anything
tag_line = tag->atts.entry.line - 1; * we match, e.g. a #if in C or C++, we check we're inside the fold level that start
* right after the tag we got from TM */
if (abs(tag_line - parent) > 1)
{
gint tag_fold = get_fold_header_after(doc->editor->sci, tag_line);
if (tag_fold >= 0)
last_child = scintilla_send_message(doc->editor->sci, SCI_GETLASTCHILD, tag_fold, -1);
}
if (line <= last_child)
{
if (tag->atts.entry.scope)
*tagname = g_strconcat(tag->atts.entry.scope,
symbols_get_context_separator(doc->file_type->id), tag->name, NULL);
else
*tagname = g_strdup(tag->name);
return tag_line; return tag_line;
} }
} }
}
/* parse the current function name here because TM line numbers may have changed, /* for the poor guy with a modified document and without real time tag parsing, we fallback
* and it would take too long to reparse the whole file. */ * to dirty and inaccurate hand-parsing */
if (doc->file_type != NULL && doc->file_type->id != GEANY_FILETYPES_NONE) else if (parent >= 0 && doc->file_type != NULL && doc->file_type->id != GEANY_FILETYPES_NONE)
{ {
const gint fn_fold = get_function_fold_number(doc); const gint fn_fold = get_function_fold_number(doc);
gint tag_line = parent;
gint fold_level = sci_get_fold_level(doc->editor->sci, tag_line);
tag_line = line; /* find the top level fold point */
do /* find the top level fold point */ while (tag_line >= 0 && (fold_level & SC_FOLDLEVELNUMBERMASK) != fn_fold)
{ {
tag_line = sci_get_fold_parent(doc->editor->sci, tag_line); tag_line = sci_get_fold_parent(doc->editor->sci, tag_line);
fold_level = sci_get_fold_level(doc->editor->sci, tag_line); fold_level = sci_get_fold_level(doc->editor->sci, tag_line);
} while (tag_line >= 0 && }
(fold_level & SC_FOLDLEVELNUMBERMASK) != fn_fold);
if (tag_line >= 0) if (tag_line >= 0)
{ {
gchar *cur_tag;
if (sci_get_lexer(doc->editor->sci) == SCLEX_CPP) if (sci_get_lexer(doc->editor->sci) == SCLEX_CPP)
cur_tag = parse_cpp_function_at_line(doc->editor->sci, tag_line); cur_tag = parse_cpp_function_at_line(doc->editor->sci, tag_line);
else else
@ -2159,13 +2158,52 @@ gint symbols_get_current_function(GeanyDocument *doc, const gchar **tagname)
} }
} }
*tagname = g_strdup(_("unknown"));
return -1;
}
static gint get_current_tag_name_cached(GeanyDocument *doc, const gchar **tagname, guint tag_types)
{
static gint tag_line = -1;
static gchar *cur_tag = NULL;
if (doc == NULL) /* reset current function */
{
current_tag_changed(NULL, -1, -1, 0);
g_free(cur_tag);
cur_tag = g_strdup(_("unknown")); cur_tag = g_strdup(_("unknown"));
if (tagname != NULL)
*tagname = cur_tag; *tagname = cur_tag;
tag_line = -1; tag_line = -1;
}
else
{
gint line = sci_get_current_line(doc->editor->sci);
gint fold_level = sci_get_fold_level(doc->editor->sci, line);
if (current_tag_changed(doc, line, fold_level, tag_types))
{
g_free(cur_tag);
tag_line = get_current_tag_name(doc, &cur_tag, tag_types);
}
*tagname = cur_tag;
}
return tag_line; return tag_line;
} }
/* Sets *tagname to point at the current function or tag name.
* If doc is NULL, reset the cached current tag data to ensure it will be reparsed on the next
* call to this function.
* Returns: line number of the current tag, or -1 if unknown. */
gint symbols_get_current_function(GeanyDocument *doc, const gchar **tagname)
{
return get_current_tag_name_cached(doc, tagname, tm_tag_function_t | tm_tag_method_t);
}
static void on_symbol_tree_sort_clicked(GtkMenuItem *menuitem, gpointer user_data) static void on_symbol_tree_sort_clicked(GtkMenuItem *menuitem, gpointer user_data)
{ {
gint sort_mode = GPOINTER_TO_INT(user_data); gint sort_mode = GPOINTER_TO_INT(user_data);