diff options
Diffstat (limited to 'src/LexAbaqus.cxx')
| -rw-r--r-- | src/LexAbaqus.cxx | 657 | 
1 files changed, 507 insertions, 150 deletions
| diff --git a/src/LexAbaqus.cxx b/src/LexAbaqus.cxx index 194b84c51..10e8b76b1 100644 --- a/src/LexAbaqus.cxx +++ b/src/LexAbaqus.cxx @@ -2,6 +2,7 @@  /** @file LexABAQUS.cxx   ** Lexer for ABAQUS. Based on the lexer for APDL by Hadar Raz.   ** By Sergio Lucato. + ** Sort of completely rewritten by Gertjan Kloosterman   **/  // The License.txt file describes the conditions under which this software may be distributed. @@ -27,7 +28,15 @@ using namespace Scintilla;  #endif  static inline bool IsAWordChar(const int ch) { -	return (ch < 0x80 && (isalnum(ch) || ch == '_')); +	return (ch < 0x80 && (isalnum(ch) || (ch == '_'))); +} + +static inline bool IsAKeywordChar(const int ch) { +	return (ch < 0x80 && (isalnum(ch) || (ch == '_') || (ch == ' '))); +} + +static inline bool IsASetChar(const int ch) { +	return (ch < 0x80 && (isalnum(ch) || (ch == '_') || (ch == '.') || (ch == '-')));  }  static inline bool IsAnOperator(char ch) { @@ -41,90 +50,228 @@ static inline bool IsAnOperator(char ch) {  	return false;  } -static void ColouriseABAQUSDoc(unsigned int startPos, int length, int initStyle, WordList *keywordlists[], +static void ColouriseABAQUSDoc(unsigned int startPos, int length, int initStyle, WordList*[] /* *keywordlists[] */,                              Accessor &styler) { - -	int stringStart = ' '; - -	WordList &processors = *keywordlists[0]; -	WordList &commands = *keywordlists[1]; -	WordList &slashcommands = *keywordlists[2]; -	WordList &starcommands = *keywordlists[3]; -	WordList &arguments = *keywordlists[4]; -	WordList &functions = *keywordlists[5]; +	enum localState { KW_LINE_KW, KW_LINE_COMMA, KW_LINE_PAR, KW_LINE_EQ, KW_LINE_VAL, \ +					  DAT_LINE_VAL, DAT_LINE_COMMA,\ +					  COMMENT_LINE,\ +					  ST_ERROR, LINE_END } state ;  	// Do not leak onto next line +	state = LINE_END ;  	initStyle = SCE_ABAQUS_DEFAULT;  	StyleContext sc(startPos, length, initStyle, styler); -	for (; sc.More(); sc.Forward()) { -		// Determine if the current state should terminate. -		if (sc.state == SCE_ABAQUS_NUMBER) { -			if (!(IsADigit(sc.ch) || sc.ch == '.' || (sc.ch == 'e' || sc.ch == 'E') || -				((sc.ch == '+' || sc.ch == '-') && (sc.chPrev == 'e' || sc.chPrev == 'E')))) { -				sc.SetState(SCE_ABAQUS_DEFAULT); -			} -		} else if (sc.state == SCE_ABAQUS_COMMENT) { -			if (sc.atLineEnd) { -				sc.SetState(SCE_ABAQUS_DEFAULT); -			} -		} else if (sc.state == SCE_ABAQUS_COMMENTBLOCK) { -			if (sc.atLineEnd) { -				if (sc.ch == '\r') { -				sc.Forward(); -				} -				sc.ForwardSetState(SCE_ABAQUS_DEFAULT); -			} -		} else if (sc.state == SCE_ABAQUS_STRING) { -			if (sc.atLineEnd) { -				sc.SetState(SCE_ABAQUS_DEFAULT); -			} else if ((sc.ch == '\'' && stringStart == '\'') || (sc.ch == '\"' && stringStart == '\"')) { -				sc.ForwardSetState(SCE_ABAQUS_DEFAULT); -			} -		} else if (sc.state == SCE_ABAQUS_WORD) { -			if (!IsAWordChar(sc.ch)) { -				char s[100]; -				sc.GetCurrentLowered(s, sizeof(s)); -				if (processors.InList(s)) { -					sc.ChangeState(SCE_ABAQUS_PROCESSOR); -				} else if (slashcommands.InList(s)) { -					sc.ChangeState(SCE_ABAQUS_SLASHCOMMAND); -				} else if (starcommands.InList(s)) { -					sc.ChangeState(SCE_ABAQUS_STARCOMMAND); -				} else if (commands.InList(s)) { -					sc.ChangeState(SCE_ABAQUS_COMMAND); -				} else if (arguments.InList(s)) { -					sc.ChangeState(SCE_ABAQUS_ARGUMENT); -				} else if (functions.InList(s)) { -					sc.ChangeState(SCE_ABAQUS_FUNCTION); -				} -				sc.SetState(SCE_ABAQUS_DEFAULT); -			} -		} else if (sc.state == SCE_ABAQUS_OPERATOR) { -			if (!IsAnOperator(static_cast<char>(sc.ch))) { -			    sc.SetState(SCE_ABAQUS_DEFAULT); -			} -		} +	// Things are actually quite simple +	// we have commentlines +	// keywordlines and datalines +	// On a data line there will only be colouring of numbers +	// a keyword line is constructed as +	// *word,[ paramname[=paramvalue]]* +	// if the line ends with a , the keyword line continues onto the new line -		// Determine if a new state should be entered. -		if (sc.state == SCE_ABAQUS_DEFAULT) { -			if (sc.ch == '*' && sc.chNext == '*') { -				sc.SetState(SCE_ABAQUS_COMMENTBLOCK); -			} else if (sc.ch == '!') { -				sc.SetState(SCE_ABAQUS_COMMENT); -			} else if (IsADigit(sc.ch) || (sc.ch == '.' && IsADigit(sc.chNext))) { -				sc.SetState(SCE_ABAQUS_NUMBER); -			} else if (sc.ch == '\'' || sc.ch == '\"') { -				sc.SetState(SCE_ABAQUS_STRING); -				stringStart = sc.ch; -			} else if (IsAWordChar(sc.ch) || ((sc.ch == '*' || sc.ch == '/') && !isgraph(sc.chPrev))) { -				sc.SetState(SCE_ABAQUS_WORD); -			} else if (IsAnOperator(static_cast<char>(sc.ch))) { -				sc.SetState(SCE_ABAQUS_OPERATOR); -			} -		} -	} -	sc.Complete(); +	for (; sc.More(); sc.Forward()) { +		switch ( state ) { +        case KW_LINE_KW : +            if ( sc.atLineEnd ) { +                // finished the line in keyword state, switch to LINE_END +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } else if ( IsAKeywordChar(sc.ch) ) { +                // nothing changes +                state = KW_LINE_KW ; +            } else if ( sc.ch == ',' ) { +                // Well well we say a comma, arguments *MUST* follow +                sc.SetState(SCE_ABAQUS_OPERATOR) ; +                state = KW_LINE_COMMA ; +            } else { +                // Flag an error +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            // Done with processing +            break ; +        case KW_LINE_COMMA : +            // acomma on a keywordline was seen +            if ( IsAKeywordChar(sc.ch)) { +                sc.SetState(SCE_ABAQUS_ARGUMENT) ; +                state = KW_LINE_PAR ; +            } else if ( sc.atLineEnd || (sc.ch == ',') ) { +                // we remain in keyword mode +                state = KW_LINE_COMMA ; +            } else if ( sc.ch == ' ' ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = KW_LINE_COMMA ; +            } else { +                // Anything else constitutes an error +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            break ; +        case KW_LINE_PAR : +            if ( sc.atLineEnd ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } else if ( IsAKeywordChar(sc.ch) || (sc.ch == '-') ) { +                // remain in this state +                state = KW_LINE_PAR ; +            } else if ( sc.ch == ',' ) { +                sc.SetState(SCE_ABAQUS_OPERATOR) ; +                state = KW_LINE_COMMA ; +            } else if ( sc.ch == '=' ) { +                sc.SetState(SCE_ABAQUS_OPERATOR) ; +                state = KW_LINE_EQ ; +            } else { +                // Anything else constitutes an error +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            break ; +        case KW_LINE_EQ : +            if ( sc.ch == ' ' ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                // remain in this state +                state = KW_LINE_EQ ; +            } else if ( IsADigit(sc.ch) || (sc.ch == '-') || (sc.ch == '.' && IsADigit(sc.chNext)) ) { +                sc.SetState(SCE_ABAQUS_NUMBER) ; +                state = KW_LINE_VAL ; +            } else if ( IsAKeywordChar(sc.ch) ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = KW_LINE_VAL ; +            } else if ( (sc.ch == '\'') || (sc.ch == '\"') ) { +                sc.SetState(SCE_ABAQUS_STRING) ; +                state = KW_LINE_VAL ; +            } else { +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            break ; +        case KW_LINE_VAL : +            if ( sc.atLineEnd ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } else if ( IsASetChar(sc.ch) && (sc.state == SCE_ABAQUS_DEFAULT) ) { +                // nothing changes +                state = KW_LINE_VAL ; +            } else if (( (IsADigit(sc.ch) || sc.ch == '.' || (sc.ch == 'e' || sc.ch == 'E') || +                    ((sc.ch == '+' || sc.ch == '-') && (sc.chPrev == 'e' || sc.chPrev == 'E')))) && +                    (sc.state == SCE_ABAQUS_NUMBER)) { +                // remain in number mode +                state = KW_LINE_VAL ; +            } else if (sc.state == SCE_ABAQUS_STRING) { +                // accept everything until a closing quote +                if ( sc.ch == '\'' || sc.ch == '\"' ) { +                    sc.SetState(SCE_ABAQUS_DEFAULT) ; +                    state = KW_LINE_VAL ; +                } +            } else if ( sc.ch == ',' ) { +                sc.SetState(SCE_ABAQUS_OPERATOR) ; +                state = KW_LINE_COMMA ; +            } else { +                // anything else is an error +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            break ; +        case DAT_LINE_VAL : +            if ( sc.atLineEnd ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } else if ( IsASetChar(sc.ch) && (sc.state == SCE_ABAQUS_DEFAULT) ) { +                // nothing changes +                state = DAT_LINE_VAL ; +            } else if (( (IsADigit(sc.ch) || sc.ch == '.' || (sc.ch == 'e' || sc.ch == 'E') || +                    ((sc.ch == '+' || sc.ch == '-') && (sc.chPrev == 'e' || sc.chPrev == 'E')))) && +                    (sc.state == SCE_ABAQUS_NUMBER)) { +                // remain in number mode +                state = DAT_LINE_VAL ; +            } else if (sc.state == SCE_ABAQUS_STRING) { +                // accept everything until a closing quote +                if ( sc.ch == '\'' || sc.ch == '\"' ) { +                    sc.SetState(SCE_ABAQUS_DEFAULT) ; +                    state = DAT_LINE_VAL ; +                } +            } else if ( sc.ch == ',' ) { +                sc.SetState(SCE_ABAQUS_OPERATOR) ; +                state = DAT_LINE_COMMA ; +            } else { +                // anything else is an error +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            break ; +        case DAT_LINE_COMMA : +            // a comma on a data line was seen +            if ( sc.atLineEnd ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } else if ( sc.ch == ' ' ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = DAT_LINE_COMMA ; +            } else if (sc.ch == ',')  { +                sc.SetState(SCE_ABAQUS_OPERATOR) ; +                state = DAT_LINE_COMMA ; +            } else if ( IsADigit(sc.ch) || (sc.ch == '-')|| (sc.ch == '.' && IsADigit(sc.chNext)) ) { +                sc.SetState(SCE_ABAQUS_NUMBER) ; +                state = DAT_LINE_VAL ; +            } else if ( IsAKeywordChar(sc.ch) ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = DAT_LINE_VAL ; +            } else if ( (sc.ch == '\'') || (sc.ch == '\"') ) { +                sc.SetState(SCE_ABAQUS_STRING) ; +                state = DAT_LINE_VAL ; +            } else { +                sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                state = ST_ERROR ; +            } +            break ; +        case COMMENT_LINE : +            if ( sc.atLineEnd ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } +            break ; +        case ST_ERROR : +            if ( sc.atLineEnd ) { +                sc.SetState(SCE_ABAQUS_DEFAULT) ; +                state = LINE_END ; +            } +            break ; +        case LINE_END : +            if ( sc.atLineEnd || sc.ch == ' ' ) { +                // nothing changes +                state = LINE_END ; +            } else if ( sc.ch == '*' ) { +                if ( sc.chNext == '*' ) { +                    state = COMMENT_LINE ; +                    sc.SetState(SCE_ABAQUS_COMMENT) ; +                } else { +                    state = KW_LINE_KW ; +                    sc.SetState(SCE_ABAQUS_STARCOMMAND) ; +                } +            } else { +                // it must be a data line, things are as if we are in DAT_LINE_COMMA +                if ( sc.ch == ',' ) { +                    sc.SetState(SCE_ABAQUS_OPERATOR) ; +                    state = DAT_LINE_COMMA ; +                } else if ( IsADigit(sc.ch) || (sc.ch == '-')|| (sc.ch == '.' && IsADigit(sc.chNext)) ) { +                    sc.SetState(SCE_ABAQUS_NUMBER) ; +                    state = DAT_LINE_VAL ; +                } else if ( IsAKeywordChar(sc.ch) ) { +                    sc.SetState(SCE_ABAQUS_DEFAULT) ; +                    state = DAT_LINE_VAL ; +                } else if ( (sc.ch == '\'') || (sc.ch == '\"') ) { +                    sc.SetState(SCE_ABAQUS_STRING) ; +                    state = DAT_LINE_VAL ; +                } else { +                    sc.SetState(SCE_ABAQUS_PROCESSOR) ; +                    state = ST_ERROR ; +                } +            } +            break ; +		  } +   } +   sc.Complete();  }  //------------------------------------------------------------------------------ @@ -166,85 +313,295 @@ static int LowerCase(int c)  	return c;  } -static int CheckABAQUSFoldPoint(char const *token, int &level) { -	if (!strcmp(token, "*step") || -		!strcmp(token, "*part") || -		!strcmp(token, "*instance") || -		!strcmp(token, "*assembly") || -		!strcmp(token, "***region") ) { -		level |= SC_FOLDLEVELHEADERFLAG; -		return 1; -	} -	if (!strcmp(token, "*end step") || -		!strcmp(token, "*end part") || -		!strcmp(token, "*end instance") || -		!strcmp(token, "*end assembly") || -		!strcmp(token, "***end region") ) { -		return -1; -	} -	return 0; +static int LineEnd(int line, Accessor &styler) +{ +    const int docLines = styler.GetLine(styler.Length() - 1);  // Available last line +    int eol_pos ; +    // if the line is the last line, the eol_pos is styler.Length() +    // eol will contain a new line, or a virtual new line +    if ( docLines == line ) +        eol_pos = styler.Length() ; +    else +        eol_pos = styler.LineStart(line + 1) - 1; +    return eol_pos ; +} + +static int LineStart(int line, Accessor &styler) +{ +    return styler.LineStart(line) ; +} + +// LineType +// +// bits determines the line type +// 1  : data line +// 2  : only whitespace +// 3  : data line with only whitespace +// 4  : keyword line +// 5  : block open keyword line +// 6  : block close keyword line +// 7  : keyword line in error +// 8  : comment line +static int LineType(int line, Accessor &styler) { +    int pos = LineStart(line, styler) ; +    int eol_pos = LineEnd(line, styler) ; + +    int c ; +    char ch = ' '; + +    int i = pos ; +    while ( i < eol_pos ) { +        c = styler.SafeGetCharAt(i); +        ch = static_cast<char>(LowerCase(c)); +        // We can say something as soon as no whitespace +        // was encountered +        if ( !IsSpace(c) ) +            break ; +        i++ ; +    } + +    if ( i >= eol_pos ) { +        // This is a whitespace line, currently +        // classifies as data line +        return 3 ; +    } + +    if ( ch != '*' ) { +        // This is a data line +        return 1 ; +    } + +    if ( i == eol_pos - 1 ) { +        // Only a single *, error but make keyword line +        return 4+3 ; +    } + +    // This means we can have a second character +    // if that is also a * this means a comment +    // otherwise it is a keyword. +    c = styler.SafeGetCharAt(i+1); +    ch = static_cast<char>(LowerCase(c)); +    if ( ch == '*' ) { +        return 8 ; +    } + +    // At this point we know this is a keyword line +    // the character at position i is a * +    // it is not a comment line +    char word[256] ; +    int  wlen = 0; + +    word[wlen] = '*' ; +	wlen++ ; + +    i++ ; +    while ( (i < eol_pos) && (wlen < 255) ) { +        c = styler.SafeGetCharAt(i); +        ch = static_cast<char>(LowerCase(c)); + +        if ( (!IsSpace(c)) && (!IsIdentifier(c)) ) +            break ; + +        if ( IsIdentifier(c) ) { +            word[wlen] = ch ; +			wlen++ ; +		} + +        i++ ; +    } + +    word[wlen] = 0 ; + +    // Make a comparison +	if ( !strcmp(word, "*step") || +         !strcmp(word, "*part") || +         !strcmp(word, "*instance") || +         !strcmp(word, "*assembly")) { +       return 4+1 ; +    } + +	if ( !strcmp(word, "*endstep") || +         !strcmp(word, "*endpart") || +         !strcmp(word, "*endinstance") || +         !strcmp(word, "*endassembly")) { +       return 4+2 ; +    } + +    return 4 ; +} + +static void SafeSetLevel(int line, int level, Accessor &styler) +{ +    if ( line < 0 ) +        return ; + +    int mask = ((~SC_FOLDLEVELHEADERFLAG) | (~SC_FOLDLEVELWHITEFLAG)); + +    if ( (level & mask) < 0 ) +        return ; + +    if ( styler.LevelAt(line) != level ) +        styler.SetLevel(line, level) ;  }  static void FoldABAQUSDoc(unsigned int startPos, int length, int, -	WordList *[], Accessor &styler) { - -	int line = styler.GetLine(startPos); -	int level = styler.LevelAt(line); -	int go = 0, done = 0; -	int endPos = startPos + length; -	char word[256]; -	int wordlen = 0; -	int i; -    bool foldCompact = styler.GetPropertyInt("fold.compact", 1) != 0; -	// Scan for tokens at the start of the line (they may include -	// whitespace, for tokens like "End Function" -	for (i = startPos; i < endPos; i++) { -		int c = styler.SafeGetCharAt(i); -		if (!done && !go) { -			if (wordlen) { // are we scanning a token already? -				word[wordlen] = static_cast<char>(LowerCase(c)); -				if (!IsIdentifier(c)) { // done with token -					word[wordlen] = '\0'; -					go = CheckABAQUSFoldPoint(word, level); -					if (!go) { -						// Treat any whitespace as single blank, for -						// things like "End   Function". -						if (IsSpace(c) && IsIdentifier(word[wordlen - 1])) { -							word[wordlen] = ' '; -							if (wordlen < 255) -								wordlen++; -						} -						else // done with this line -							done = 1; -					} -				} else if (wordlen < 255) { -					wordlen++; +WordList *[], Accessor &styler) { +    int startLine = styler.GetLine(startPos) ; +    int endLine   = styler.GetLine(startPos+length-1) ; + +    // bool foldCompact = styler.GetPropertyInt("fold.compact", 1) != 0; +    // We want to deal with all the cases +    // To know the correct indentlevel, we need to look back to the +    // previous command line indentation level +	// order of formatting keyline datalines commentlines +    int beginData    = -1 ; +    int beginComment = -1 ; +    int prvKeyLine   = startLine ; +    int prvKeyLineTp =  0 ; + +    // Scan until we find the previous keyword line +    // this will give us the level reference that we need +    while ( prvKeyLine > 0 ) { +        prvKeyLine-- ; +        prvKeyLineTp = LineType(prvKeyLine, styler) ; +        if ( prvKeyLineTp & 4 ) +            break ; +    } + +    // Determine the base line level of all lines following +    // the previous keyword +    // new keyword lines are placed on this level +    //if ( prvKeyLineTp & 4 ) { +    int level = styler.LevelAt(prvKeyLine) & ~SC_FOLDLEVELHEADERFLAG ; +    //} + +    // uncomment line below if weird behaviour continues +    prvKeyLine = -1 ; + +    // Now start scanning over the lines. +    for ( int line = startLine; line <= endLine; line++ ) { +        int lineType = LineType(line, styler) ; + +        // Check for comment line +        if ( lineType == 8 ) { +            if ( beginComment < 0 ) { +                beginComment = line ; +			} +        } + +        // Check for data line +        if ( (lineType == 1) || (lineType == 3) ) { +            if ( beginData < 0 ) { +                if ( beginComment >= 0 ) { +                    beginData = beginComment ; +                } else { +                    beginData = line ; +                } +            } +			beginComment = -1 ; +		} + +        // Check for keywordline. +        // As soon as a keyword line is encountered, we can set the +        // levels of everything from the previous keyword line to this one +        if ( lineType & 4 ) { +            // this is a keyword, we can now place the previous keyword +            // all its data lines and the remainder + +            // Write comments and data line +            if ( beginComment < 0 ) { +                beginComment = line ; +			} + +            if ( beginData < 0 ) { +                beginData = beginComment ; +				if ( prvKeyLineTp != 5 ) +					SafeSetLevel(prvKeyLine, level, styler) ; +				else +					SafeSetLevel(prvKeyLine, level | SC_FOLDLEVELHEADERFLAG, styler) ; +            } else { +                SafeSetLevel(prvKeyLine, level | SC_FOLDLEVELHEADERFLAG, styler) ; +            } + +            int datLevel = level + 1 ; +			if ( !(prvKeyLineTp & 4) ) { +				datLevel = level ; +			} + +            for ( int ll = beginData; ll < beginComment; ll++ ) +                SafeSetLevel(ll, datLevel, styler) ; + +            // The keyword we just found is going to be written at another level +            // if we have a type 5 and type 6 +            if ( prvKeyLineTp == 5 ) { +                level += 1 ; +			} + +            if ( prvKeyLineTp == 6 ) { +                level -= 1 ; +				if ( level < 0 ) { +					level = 0 ;  				} -			} else { // start scanning at first non-whitespace character -				if (!IsSpace(c)) { -					if (IsIdentifier(c)) { -						word[0] = static_cast<char>(LowerCase(c)); -						wordlen = 1; -					} else // done with this line -						done = 1; +            } + +            for ( int lll = beginComment; lll < line; lll++ ) +                SafeSetLevel(lll, level, styler) ; + +            // wrap and reset +            beginComment = -1 ; +            beginData    = -1 ; +            prvKeyLine   = line ; +            prvKeyLineTp = lineType ; +        } + +    } + +    if ( beginComment < 0 ) { +        beginComment = endLine + 1 ; +    } else { +        // We need to find out whether this comment block is followed by +        // a data line or a keyword line +        const int docLines = styler.GetLine(styler.Length() - 1); + +        for ( int line = endLine + 1; line <= docLines; line++ ) { +            int lineType = LineType(line, styler) ; + +            if ( lineType != 8 ) { +				if ( !(lineType & 4) )  { +					beginComment = endLine + 1 ;  				} +                break ;  			} -		} -		if (c == '\n') { // line end -			if (!done && wordlen == 0 && foldCompact) // line was only space -				level |= SC_FOLDLEVELWHITEFLAG; -			if (level != styler.LevelAt(line)) -				styler.SetLevel(line, level); -			level += go; -			line++; -			// reset state -			wordlen = 0; -			level &= ~SC_FOLDLEVELHEADERFLAG; -			level &= ~SC_FOLDLEVELWHITEFLAG; -			go = 0; -			done = 0; -		} +        } +    } + +    if ( beginData < 0 ) { +        beginData = beginComment ; +		if ( prvKeyLineTp != 5 ) +			SafeSetLevel(prvKeyLine, level, styler) ; +		else +			SafeSetLevel(prvKeyLine, level | SC_FOLDLEVELHEADERFLAG, styler) ; +    } else { +        SafeSetLevel(prvKeyLine, level | SC_FOLDLEVELHEADERFLAG, styler) ; +    } + +    int datLevel = level + 1 ; +	if ( !(prvKeyLineTp & 4) ) { +		datLevel = level ; +	} + +    for ( int ll = beginData; ll < beginComment; ll++ ) +        SafeSetLevel(ll, datLevel, styler) ; + +	if ( prvKeyLineTp == 5 ) { +		level += 1 ; +	} + +	if ( prvKeyLineTp == 6 ) { +		level -= 1 ;  	} +	for ( int m = beginComment; m <= endLine; m++ ) +        SafeSetLevel(m, level, styler) ;  }  static const char * const abaqusWordListDesc[] = { | 
