improved parser to detect variables and members

git-svn-id: https://geany.svn.sourceforge.net/svnroot/geany/trunk@327 ea778897-0a13-0410-b9d1-a72fbfd435f5
This commit is contained in:
Enrico Tröger 2006-05-12 23:23:52 +00:00
parent 138f80a69f
commit 88e763b9cc
2 changed files with 165 additions and 84 deletions

View File

@ -117,6 +117,27 @@ void treeviews_init_tag_list(gint idx)
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_namespace), 0, _("Begin"), -1); gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_namespace), 0, _("Begin"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_other), NULL); gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_other), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_other), 0, _("Other"), -1); gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_other), 0, _("Other"), -1);
*/
break;
}
case GEANY_FILETYPES_RUBY:
{
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_function), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_function), 0, _("Methods"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_class), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_class), 0, _("Class"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_member), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_member), 0, _("Singleton"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_macro), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_macro), 0, _("Mixin"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_variable), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_variable), 0, _("Variables"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_struct), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_struct), 0, _("Members"), -1);
/* gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_namespace), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_namespace), 0, _("Begin"), -1);
gtk_tree_store_append(doc_list[idx].tag_store, &(tv.tag_other), NULL);
gtk_tree_store_set(doc_list[idx].tag_store, &(tv.tag_other), 0, _("Other"), -1);
*/ */
break; break;
} }

View File

@ -1,4 +1,5 @@
/* /*
* $Id: ruby.c,v 1.2 2001/12/18 04:30:18 darren Exp $
* *
* Copyright (c) 2000-2001, Thaddeus Covert <sahuagin@mediaone.net> * Copyright (c) 2000-2001, Thaddeus Covert <sahuagin@mediaone.net>
* *
@ -7,6 +8,10 @@
* *
* This module contains functions for generating tags for Ruby language * This module contains functions for generating tags for Ruby language
* files. * files.
*
* Copyright (c) 2002, Matthias Veit <matthias_veit@yahoo.de>
* Enable parsing of class (in ruby: singleton) methods and mixins.
*
*/ */
/* /*
@ -24,111 +29,166 @@
* DATA DEFINITIONS * DATA DEFINITIONS
*/ */
typedef enum { typedef enum {
K_CLASS, K_FUNCTION K_CLASS, K_METHOD, K_SINGLETON, K_MIXIN, K_VARIABLE, K_MEMBER
} rubyKind; } rubyKind;
static kindOption RubyKinds [] = { static kindOption RubyKinds [] = {
{ TRUE, 'c', "class", "classes" }, { TRUE, 'c', "class", "classes" },
{ TRUE, 'f', "function", "functions" }, { TRUE, 'f', "function", "methods" },
{ TRUE, 'm', "mixin", "mixins" } { TRUE, 'm', "member", "singleton_methods" },
}; { TRUE, 'd', "macro", "mixins" },
{ TRUE, 'v', "variable", "variable" },
{ TRUE, 's', "struct", "member" }
};
/* /*
* FUNCTION DEFINITIONS * FUNCTION DEFINITIONS
*/ */
static void findRubyTags (void) static void findRubyTags (void) {
{
vString *name = vStringNew (); vString *name = vStringNew ();
const unsigned char *line; const unsigned char *line;
boolean inMultilineString = FALSE; boolean inMultilineString = FALSE;
while ((line = fileReadLine ()) != NULL) while ((line = fileReadLine ()) != NULL) {
{ const unsigned char *cp = line;
const unsigned char *cp = line; boolean is_singleton = FALSE;
while (*cp != '\0') while (*cp != '\0')
{ {
if (*cp=='"' && if (*cp=='"' &&
strncmp ((const char*) cp, "\"\"\"", (size_t) 3) == 0) strncmp ((const char*) cp, "\"\"\"", (size_t) 3) == 0) {
{ inMultilineString = (boolean) !inMultilineString;
inMultilineString = (boolean) !inMultilineString; cp += 3;
cp += 3; }
} /* make sure you include the comments */
/* make sure you include the comments */ if(*cp=='=' && strncmp((const char*)cp, "==begin", (size_t)7) == 0) {
if(*cp=='=' && strncmp((const char*)cp, "==begin", (size_t)7) == 0) inMultilineString = (boolean)!inMultilineString;
{ cp +=3;
inMultilineString = (boolean)!inMultilineString; }
cp +=3; /* mark the end of a comment */
} if( *cp=='=' && strncmp((const char*)cp, "==end", (size_t)5) == 0) {
/* mark the end of a comment */ inMultilineString = (boolean)0;
if( *cp=='=' && strncmp((const char*)cp, "==end", (size_t)5) == 0) cp+=5;
{ }
inMultilineString = (boolean)0; if (inMultilineString || isspace ((int) *cp))
cp+=5; ++cp;
} else if (*cp == '#')
if (inMultilineString || isspace ((int) *cp)) break;
++cp; else if (*cp == '=' && *(cp+1) != '=' && (isspace((int) *(cp-1)) || isalnum((int) *(cp-1)))) {
else if (*cp == '#')
break;
else if (strncmp ((const char*) cp, "class", (size_t) 5) == 0)
{
cp += 5;
if (isspace ((int) *cp))
{
while (isspace ((int) *cp))
++cp;
while (isalnum ((int) *cp) || *cp == '_' || *cp == ':')
{
vStringPut (name, (int) *cp);
++cp;
}
vStringTerminate (name);
makeSimpleTag (name, RubyKinds, K_CLASS);
vStringClear (name);
}
}
else if (strncmp ((const char*) cp, "def", (size_t) 3) == 0)
{
cp += 3;
if (isspace ((int) *cp))
{
while (isalnum ((int) *cp) || *cp == ':')
++cp;
/* Put the valid characters allowed in a variable name // try to detect a variable by the = sign - enrico
* in here. In ruby a variable name ENDING in ! means // exlude all what not look like ' = ' or 'l=b'
* it changes the underlying data structure in place. const unsigned char *cur_pos = cp; // store the current position because
* A variable name ENDING in ? means that the function // we are going backwards with cp
* returns a bool. Functions should not start with these // (think about what happens if we don't do this ;-))
* characters. int is_member = 0;
*/ cp--;
while (isalnum ((int) *cp) || *cp == '_' ||
*cp == '!' || *cp =='?') while (isspace ((int) *cp)) --cp;
{
vStringPut (name, (int) *cp); while (isalnum ((int) *cp) || *cp == '_') cp--;
++cp;
} if (*cp == '@') is_member = 1; // handle @...
vStringTerminate (name); else if (!isspace((int) *cp))
makeSimpleTag (name, RubyKinds, K_FUNCTION); {
vStringClear (name); cp = cur_pos + 1;
} continue;
} }
else if (*cp != '\0')
{ // cp points to the char before the variable name, so go forward
do cp++;
++cp;
while (isalnum ((int) *cp) || *cp == '_'); while (cp != cur_pos && ! isspace((int) *cp))
} {
} vStringPut (name, (int) *cp);
cp++;
}
vStringTerminate (name);
if (vStringLength (name) > 0)
{
if (is_member) makeSimpleTag (name, RubyKinds, K_MEMBER);
else makeSimpleTag (name, RubyKinds, K_VARIABLE);
}
vStringClear (name);
cp = cur_pos + 1;
}
else if (strncmp ((const char*) cp, "module", (size_t) 6) == 0) {
cp += 6;
if (isspace ((int) *cp)) {
while (isspace ((int) *cp))
++cp;
while (isalnum ((int) *cp) || *cp == '_' || *cp == ':') {
vStringPut (name, (int) *cp);
++cp;
}
vStringTerminate (name);
makeSimpleTag (name, RubyKinds, K_MIXIN);
vStringClear (name);
}
} else if (strncmp ((const char*) cp, "class", (size_t) 5) == 0) {
cp += 5;
if (isspace ((int) *cp)) {
while (isspace ((int) *cp))
++cp;
while (isalnum ((int) *cp) || *cp == '_' || *cp == ':') {
vStringPut (name, (int) *cp);
++cp;
}
vStringTerminate (name);
makeSimpleTag (name, RubyKinds, K_CLASS);
vStringClear (name);
}
} else if (strncmp ((const char*) cp, "def", (size_t) 3) == 0) {
cp += 3;
if (isspace ((int) *cp)) {
while (isspace ((int) *cp))
++cp;
/* Put the valid characters allowed in a variable name
* in here. In ruby a variable name ENDING in ! means
* it changes the underlying data structure in place.
* A variable name ENDING in ? means that the function
* returns a bool. Functions should not start with these
* characters.
*/
while (isalnum ((int) *cp) || *cp == '_' || *cp == '!' || *cp =='?' || *cp=='.') {
/* classmethods are accesible only via class instance instead
* of object instance. This difference has to be outlined.
*/
if (*cp == '.') {
//class method
is_singleton = TRUE;
vStringTerminate (name);
vStringClear(name);
} else {
vStringPut (name, (int) *cp);
}
++cp;
}
vStringTerminate (name);
if (is_singleton) {
makeSimpleTag (name, RubyKinds, K_SINGLETON);
} else {
makeSimpleTag (name, RubyKinds, K_METHOD);
}
vStringClear (name);
}
} else if (*cp != '\0') {
do
++cp;
while (isalnum ((int) *cp) || *cp == '_');
}
}
} }
vStringDelete (name); vStringDelete (name);
} }
extern parserDefinition* RubyParser (void) extern parserDefinition* RubyParser (void)
{ {
static const char *const extensions [] = { "rb", NULL }; static const char *const extensions [] = { "rb", "rhtml", NULL };
parserDefinition* def = parserNew ("Ruby"); parserDefinition* def = parserNew ("Ruby");
def->kinds = RubyKinds; def->kinds = RubyKinds;
def->kindCount = KIND_COUNT (RubyKinds); def->kindCount = KIND_COUNT (RubyKinds);