forked from qt-creator/qt-creator
		
	QmlJS: Update parser from Qt5.
Change-Id: I26a5fb6a1eb99a777e4f1e01fb7b19e559c9bce7 Reviewed-on: http://codereview.qt-project.org/5143 Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
This commit is contained in:
		| @@ -10,10 +10,6 @@ for i in $QTDIR/src/declarative/qml/qdeclarative{error.{h,cpp},dirparser{_p.h,.c | |||||||
|     sed -f $me/cmd.sed $i > $me/$(echo $(basename $i) | sed s/qdeclarative/qml/) |     sed -f $me/cmd.sed $i > $me/$(echo $(basename $i) | sed s/qdeclarative/qml/) | ||||||
| done | done | ||||||
|  |  | ||||||
| for i in $QTDIR/src/declarative/qml/ftw/qdeclarativeutils_p.h; do |  | ||||||
|     sed -f $me/cmd.sed $i > $me/$(echo $(basename $i) | sed s/qdeclarative/qml/) |  | ||||||
| done |  | ||||||
|  |  | ||||||
| # export QmlDirParser | # export QmlDirParser | ||||||
| perl -p -0777 -i -e 's/QT_BEGIN_NAMESPACE\n\nclass QmlError;\nclass QmlDirParser/#include "qmljsglobal_p.h"\n\nQT_BEGIN_NAMESPACE\n\nclass QmlError;\nclass QML_PARSER_EXPORT QmlDirParser/' qmldirparser_p.h | perl -p -0777 -i -e 's/QT_BEGIN_NAMESPACE\n\nclass QmlError;\nclass QmlDirParser/#include "qmljsglobal_p.h"\n\nQT_BEGIN_NAMESPACE\n\nclass QmlError;\nclass QML_PARSER_EXPORT QmlDirParser/' qmldirparser_p.h | ||||||
| # replace qmlglobal_p.h include with needed declaration | # replace qmlglobal_p.h include with needed declaration | ||||||
|   | |||||||
| @@ -33,7 +33,6 @@ | |||||||
| #include "qmldirparser_p.h" | #include "qmldirparser_p.h" | ||||||
| #include "qmlerror.h" | #include "qmlerror.h" | ||||||
| bool Qml_isFileCaseCorrect(const QString &) { return true; } | bool Qml_isFileCaseCorrect(const QString &) { return true; } | ||||||
| #include <qmlutils_p.h> |  | ||||||
|  |  | ||||||
| #include <QtCore/QTextStream> | #include <QtCore/QTextStream> | ||||||
| #include <QtCore/QFile> | #include <QtCore/QFile> | ||||||
| @@ -132,9 +131,9 @@ bool QmlDirParser::parse() | |||||||
|         while (index != length) { |         while (index != length) { | ||||||
|             const QChar ch = line.at(index); |             const QChar ch = line.at(index); | ||||||
|  |  | ||||||
|             if (QmlUtils::isSpace(ch)) { |             if (ch.isSpace()) { | ||||||
|                 do { ++index; } |                 do { ++index; } | ||||||
|                 while (index != length && QmlUtils::isSpace(line.at(index))); |                 while (index != length && line.at(index).isSpace()); | ||||||
|  |  | ||||||
|             } else if (ch == QLatin1Char('#')) { |             } else if (ch == QLatin1Char('#')) { | ||||||
|                 // recognized a comment |                 // recognized a comment | ||||||
| @@ -144,7 +143,7 @@ bool QmlDirParser::parse() | |||||||
|                 const int start = index; |                 const int start = index; | ||||||
|  |  | ||||||
|                 do { ++index; } |                 do { ++index; } | ||||||
|                 while (index != length && !QmlUtils::isSpace(line.at(index))); |                 while (index != length && !line.at(index).isSpace()); | ||||||
|  |  | ||||||
|                 const QString lexeme = line.mid(start, index - start); |                 const QString lexeme = line.mid(start, index - start); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -125,7 +125,7 @@ | |||||||
| **************************************************************************/ | **************************************************************************/ | ||||||
|  |  | ||||||
| #include <QtCore/QtDebug> | #include <QtCore/QtDebug> | ||||||
| #include <QtGui/QApplication> | #include <QtCore/QCoreApplication> | ||||||
|  |  | ||||||
| #include <string.h> | #include <string.h> | ||||||
|  |  | ||||||
| @@ -398,14 +398,6 @@ void Parser::reallocateStack() | |||||||
|     string_stack = reinterpret_cast<QStringRef*> (qRealloc(string_stack, stack_size * sizeof(QStringRef))); |     string_stack = reinterpret_cast<QStringRef*> (qRealloc(string_stack, stack_size * sizeof(QStringRef))); | ||||||
| } | } | ||||||
|  |  | ||||||
| inline static bool automatic(Engine *driver, int token) |  | ||||||
| { |  | ||||||
|     return token == $table::T_RBRACE |  | ||||||
|         || token == 0 |  | ||||||
|         || driver->lexer()->prevTerminator(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Parser::Parser(Engine *engine): | Parser::Parser(Engine *engine): | ||||||
|     driver(engine), |     driver(engine), | ||||||
|     pool(engine->pool()), |     pool(engine->pool()), | ||||||
| @@ -2881,7 +2873,7 @@ PropertyNameAndValueListOpt: PropertyNameAndValueList ; | |||||||
|         const int errorState = state_stack[tos]; |         const int errorState = state_stack[tos]; | ||||||
|  |  | ||||||
|         // automatic insertion of `;' |         // automatic insertion of `;' | ||||||
|         if (yytoken != -1 && t_action(errorState, T_AUTOMATIC_SEMICOLON) && automatic(driver, yytoken)) { |         if (yytoken != -1 && t_action(errorState, T_AUTOMATIC_SEMICOLON) && lexer->canInsertAutomaticSemicolon(yytoken)) { | ||||||
|             SavedToken &tk = token_buffer[0]; |             SavedToken &tk = token_buffer[0]; | ||||||
|             tk.token = yytoken; |             tk.token = yytoken; | ||||||
|             tk.dval = yylval; |             tk.dval = yylval; | ||||||
|   | |||||||
| @@ -38,37 +38,18 @@ | |||||||
| #include <QtCore/QVarLengthArray> | #include <QtCore/QVarLengthArray> | ||||||
| #include <QtCore/QDebug> | #include <QtCore/QDebug> | ||||||
|  |  | ||||||
| #ifndef QMLJS_NO_FTW |  | ||||||
| #include <qmlutils_p.h> |  | ||||||
| #else |  | ||||||
| namespace QmlUtils { |  | ||||||
| inline bool isUpper(const QChar &qc){ return qc.isUpper(); } |  | ||||||
| inline bool isLower(const QChar &qc) { return qc.isLower(); } |  | ||||||
| inline bool isLetter(const QChar &qc) { return qc.isLetter(); } |  | ||||||
| inline bool isDigit(const QChar &qc) { return qc.isDigit(); } |  | ||||||
| inline bool isLetterOrNumber(const QChar &qc) { return qc.isLetterOrNumber(); } |  | ||||||
| inline bool isSpace(const QChar &qc) { return qc.isSpace(); } |  | ||||||
| } // namespace QmlUtils |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| QT_BEGIN_NAMESPACE | QT_BEGIN_NAMESPACE | ||||||
| Q_CORE_EXPORT double qstrtod(const char *s00, char const **se, bool *ok); | Q_CORE_EXPORT double qstrtod(const char *s00, char const **se, bool *ok); | ||||||
| QT_END_NAMESPACE | QT_END_NAMESPACE | ||||||
|  |  | ||||||
| using namespace QmlJS; | using namespace QmlJS; | ||||||
|  |  | ||||||
| enum RegExpFlag { | static int regExpFlagFromChar(const QChar &ch) | ||||||
|     Global     = 0x01, |  | ||||||
|     IgnoreCase = 0x02, |  | ||||||
|     Multiline  = 0x04 |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| static int flagFromChar(const QChar &ch) |  | ||||||
| { | { | ||||||
|     switch (ch.unicode()) { |     switch (ch.unicode()) { | ||||||
|     case 'g': return Global; |     case 'g': return Lexer::RegExp_Global; | ||||||
|     case 'i': return IgnoreCase; |     case 'i': return Lexer::RegExp_IgnoreCase; | ||||||
|     case 'm': return Multiline; |     case 'm': return Lexer::RegExp_Multiline; | ||||||
|     } |     } | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
| @@ -115,6 +96,7 @@ Lexer::Lexer(Engine *engine) | |||||||
|     , _prohibitAutomaticSemicolon(false) |     , _prohibitAutomaticSemicolon(false) | ||||||
|     , _restrictedKeyword(false) |     , _restrictedKeyword(false) | ||||||
|     , _terminator(false) |     , _terminator(false) | ||||||
|  |     , _followsClosingBrace(false) | ||||||
|     , _delimited(false) |     , _delimited(false) | ||||||
|     , _qmlMode(true) |     , _qmlMode(true) | ||||||
| { | { | ||||||
| @@ -164,6 +146,7 @@ void Lexer::setCode(const QString &code, int lineno, bool qmlMode) | |||||||
|     _prohibitAutomaticSemicolon = false; |     _prohibitAutomaticSemicolon = false; | ||||||
|     _restrictedKeyword = false; |     _restrictedKeyword = false; | ||||||
|     _terminator = false; |     _terminator = false; | ||||||
|  |     _followsClosingBrace = false; | ||||||
|     _delimited = false; |     _delimited = false; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -179,12 +162,15 @@ void Lexer::scanChar() | |||||||
|  |  | ||||||
| int Lexer::lex() | int Lexer::lex() | ||||||
| { | { | ||||||
|  |     const int previousTokenKind = _tokenKind; | ||||||
|  |  | ||||||
|     _tokenSpell = QStringRef(); |     _tokenSpell = QStringRef(); | ||||||
|     _tokenKind = scanToken(); |     _tokenKind = scanToken(); | ||||||
|     _tokenLength = _codePtr - _tokenStartPtr - 1; |     _tokenLength = _codePtr - _tokenStartPtr - 1; | ||||||
|  |  | ||||||
|     _delimited = false; |     _delimited = false; | ||||||
|     _restrictedKeyword = false; |     _restrictedKeyword = false; | ||||||
|  |     _followsClosingBrace = (previousTokenKind == T_RBRACE); | ||||||
|  |  | ||||||
|     // update the flags |     // update the flags | ||||||
|     switch (_tokenKind) { |     switch (_tokenKind) { | ||||||
| @@ -285,7 +271,7 @@ again: | |||||||
|     _validTokenText = false; |     _validTokenText = false; | ||||||
|     _tokenLinePtr = _lastLinePtr; |     _tokenLinePtr = _lastLinePtr; | ||||||
|  |  | ||||||
|     while (QmlUtils::isSpace(_char)) { |     while (_char.isSpace()) { | ||||||
|         if (_char == QLatin1Char('\n')) { |         if (_char == QLatin1Char('\n')) { | ||||||
|             _tokenLinePtr = _codePtr; |             _tokenLinePtr = _codePtr; | ||||||
|  |  | ||||||
| @@ -424,19 +410,19 @@ again: | |||||||
|         return T_DIVIDE_; |         return T_DIVIDE_; | ||||||
|  |  | ||||||
|     case '.': |     case '.': | ||||||
|         if (QmlUtils::isDigit(_char)) { |         if (_char.isDigit()) { | ||||||
|             QVarLengthArray<char,32> chars; |             QVarLengthArray<char,32> chars; | ||||||
|  |  | ||||||
|             chars.append(ch.unicode()); // append the `.' |             chars.append(ch.unicode()); // append the `.' | ||||||
|  |  | ||||||
|             while (QmlUtils::isDigit(_char)) { |             while (_char.isDigit()) { | ||||||
|                 chars.append(_char.unicode()); |                 chars.append(_char.unicode()); | ||||||
|                 scanChar(); |                 scanChar(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (_char == QLatin1Char('e') || _char == QLatin1Char('E')) { |             if (_char == QLatin1Char('e') || _char == QLatin1Char('E')) { | ||||||
|                 if (QmlUtils::isDigit(_codePtr[0]) || ((_codePtr[0] == QLatin1Char('+') || _codePtr[0] == QLatin1Char('-')) && |                 if (_codePtr[0].isDigit() || ((_codePtr[0] == QLatin1Char('+') || _codePtr[0] == QLatin1Char('-')) && | ||||||
|                                               QmlUtils::isDigit(_codePtr[1]))) { |                                               _codePtr[1].isDigit())) { | ||||||
|  |  | ||||||
|                     chars.append(_char.unicode()); |                     chars.append(_char.unicode()); | ||||||
|                     scanChar(); // consume `e' |                     scanChar(); // consume `e' | ||||||
| @@ -446,7 +432,7 @@ again: | |||||||
|                         scanChar(); // consume the sign |                         scanChar(); // consume the sign | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     while (QmlUtils::isDigit(_char)) { |                     while (_char.isDigit()) { | ||||||
|                         chars.append(_char.unicode()); |                         chars.append(_char.unicode()); | ||||||
|                         scanChar(); |                         scanChar(); | ||||||
|                     } |                     } | ||||||
| @@ -672,7 +658,7 @@ again: | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     default: |     default: | ||||||
|         if (QmlUtils::isLetter(ch) || ch == QLatin1Char('$') || ch == QLatin1Char('_') || (ch == QLatin1Char('\\') && _char == QLatin1Char('u'))) { |         if (ch.isLetter() || ch == QLatin1Char('$') || ch == QLatin1Char('_') || (ch == QLatin1Char('\\') && _char == QLatin1Char('u'))) { | ||||||
|             bool identifierWithEscapeChars = false; |             bool identifierWithEscapeChars = false; | ||||||
|             if (ch == QLatin1Char('\\')) { |             if (ch == QLatin1Char('\\')) { | ||||||
|                 identifierWithEscapeChars = true; |                 identifierWithEscapeChars = true; | ||||||
| @@ -687,7 +673,7 @@ again: | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             while (true) { |             while (true) { | ||||||
|                 if (QmlUtils::isLetterOrNumber(_char) || _char == QLatin1Char('$') || _char == QLatin1Char('_')) { |                 if (_char.isLetterOrNumber() || _char == QLatin1Char('$') || _char == QLatin1Char('_')) { | ||||||
|                     if (identifierWithEscapeChars) |                     if (identifierWithEscapeChars) | ||||||
|                         _tokenText += _char; |                         _tokenText += _char; | ||||||
|  |  | ||||||
| @@ -726,13 +712,13 @@ again: | |||||||
|                     return kind; |                     return kind; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } else if (QmlUtils::isDigit(ch)) { |         } else if (ch.isDigit()) { | ||||||
|             if (ch != QLatin1Char('0')) { |             if (ch != QLatin1Char('0')) { | ||||||
|                 double integer = ch.unicode() - '0'; |                 double integer = ch.unicode() - '0'; | ||||||
|  |  | ||||||
|                 QChar n = _char; |                 QChar n = _char; | ||||||
|                 const QChar *code = _codePtr; |                 const QChar *code = _codePtr; | ||||||
|                 while (QmlUtils::isDigit(n)) { |                 while (n.isDigit()) { | ||||||
|                     integer = integer * 10 + (n.unicode() - '0'); |                     integer = integer * 10 + (n.unicode() - '0'); | ||||||
|                     n = *code++; |                     n = *code++; | ||||||
|                 } |                 } | ||||||
| @@ -766,7 +752,7 @@ again: | |||||||
|             } |             } | ||||||
|  |  | ||||||
|             // decimal integer literal |             // decimal integer literal | ||||||
|             while (QmlUtils::isDigit(_char)) { |             while (_char.isDigit()) { | ||||||
|                 chars.append(_char.unicode()); |                 chars.append(_char.unicode()); | ||||||
|                 scanChar(); // consume the digit |                 scanChar(); // consume the digit | ||||||
|             } |             } | ||||||
| @@ -775,14 +761,14 @@ again: | |||||||
|                 chars.append(_char.unicode()); |                 chars.append(_char.unicode()); | ||||||
|                 scanChar(); // consume `.' |                 scanChar(); // consume `.' | ||||||
|  |  | ||||||
|                 while (QmlUtils::isDigit(_char)) { |                 while (_char.isDigit()) { | ||||||
|                     chars.append(_char.unicode()); |                     chars.append(_char.unicode()); | ||||||
|                     scanChar(); |                     scanChar(); | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 if (_char == QLatin1Char('e') || _char == QLatin1Char('E')) { |                 if (_char == QLatin1Char('e') || _char == QLatin1Char('E')) { | ||||||
|                     if (QmlUtils::isDigit(_codePtr[0]) || ((_codePtr[0] == QLatin1Char('+') || _codePtr[0] == QLatin1Char('-')) && |                     if (_codePtr[0].isDigit() || ((_codePtr[0] == QLatin1Char('+') || _codePtr[0] == QLatin1Char('-')) && | ||||||
|                                                   QmlUtils::isDigit(_codePtr[1]))) { |                                                   _codePtr[1].isDigit())) { | ||||||
|  |  | ||||||
|                         chars.append(_char.unicode()); |                         chars.append(_char.unicode()); | ||||||
|                         scanChar(); // consume `e' |                         scanChar(); // consume `e' | ||||||
| @@ -792,15 +778,15 @@ again: | |||||||
|                             scanChar(); // consume the sign |                             scanChar(); // consume the sign | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         while (QmlUtils::isDigit(_char)) { |                         while (_char.isDigit()) { | ||||||
|                             chars.append(_char.unicode()); |                             chars.append(_char.unicode()); | ||||||
|                             scanChar(); |                             scanChar(); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } else if (_char == QLatin1Char('e') || _char == QLatin1Char('E')) { |             } else if (_char == QLatin1Char('e') || _char == QLatin1Char('E')) { | ||||||
|                 if (QmlUtils::isDigit(_codePtr[0]) || ((_codePtr[0] == QLatin1Char('+') || _codePtr[0] == QLatin1Char('-')) && |                 if (_codePtr[0].isDigit() || ((_codePtr[0] == QLatin1Char('+') || _codePtr[0] == QLatin1Char('-')) && | ||||||
|                                               QmlUtils::isDigit(_codePtr[1]))) { |                                               _codePtr[1].isDigit())) { | ||||||
|  |  | ||||||
|                     chars.append(_char.unicode()); |                     chars.append(_char.unicode()); | ||||||
|                     scanChar(); // consume `e' |                     scanChar(); // consume `e' | ||||||
| @@ -810,7 +796,7 @@ again: | |||||||
|                         scanChar(); // consume the sign |                         scanChar(); // consume the sign | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     while (QmlUtils::isDigit(_char)) { |                     while (_char.isDigit()) { | ||||||
|                         chars.append(_char.unicode()); |                         chars.append(_char.unicode()); | ||||||
|                         scanChar(); |                         scanChar(); | ||||||
|                     } |                     } | ||||||
| @@ -862,7 +848,7 @@ bool Lexer::scanRegExp(RegExpBodyPrefix prefix) | |||||||
|             // scan the flags |             // scan the flags | ||||||
|             _patternFlags = 0; |             _patternFlags = 0; | ||||||
|             while (isIdentLetter(_char)) { |             while (isIdentLetter(_char)) { | ||||||
|                 int flag = flagFromChar(_char); |                 int flag = regExpFlagFromChar(_char); | ||||||
|                 if (flag == 0) { |                 if (flag == 0) { | ||||||
|                     _errorMessage = QCoreApplication::translate("QmlParser", "Invalid regular expression flag '%0'") |                     _errorMessage = QCoreApplication::translate("QmlParser", "Invalid regular expression flag '%0'") | ||||||
|                              .arg(QChar(_char)); |                              .arg(QChar(_char)); | ||||||
| @@ -969,6 +955,11 @@ bool Lexer::isOctalDigit(ushort c) | |||||||
|     return (c >= '0' && c <= '7'); |     return (c >= '0' && c <= '7'); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | int Lexer::tokenKind() const | ||||||
|  | { | ||||||
|  |     return _tokenKind; | ||||||
|  | } | ||||||
|  |  | ||||||
| int Lexer::tokenOffset() const | int Lexer::tokenOffset() const | ||||||
| { | { | ||||||
|     return _tokenStartPtr - _code.unicode(); |     return _tokenStartPtr - _code.unicode(); | ||||||
| @@ -1048,4 +1039,113 @@ bool Lexer::prevTerminator() const | |||||||
|     return _terminator; |     return _terminator; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | bool Lexer::followsClosingBrace() const | ||||||
|  | { | ||||||
|  |     return _followsClosingBrace; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Lexer::canInsertAutomaticSemicolon(int token) const | ||||||
|  | { | ||||||
|  |     return token == T_RBRACE | ||||||
|  |             || token == EOF_SYMBOL | ||||||
|  |             || _terminator | ||||||
|  |             || _followsClosingBrace; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Lexer::scanDirectives(Directives *directives) | ||||||
|  | { | ||||||
|  |     if (_qmlMode) { | ||||||
|  |         // the directives are a Javascript-only extension. | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     lex(); // fetch the first token | ||||||
|  |  | ||||||
|  |     if (_tokenKind != T_DOT) | ||||||
|  |         return true; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |         lex(); // skip T_DOT | ||||||
|  |  | ||||||
|  |         const int lineNumber = tokenStartLine(); | ||||||
|  |  | ||||||
|  |         if (! (_tokenKind == T_IDENTIFIER || _tokenKind == T_RESERVED_WORD)) | ||||||
|  |             return false; // expected a valid QML/JS directive | ||||||
|  |  | ||||||
|  |         const QString directiveName = tokenText(); | ||||||
|  |  | ||||||
|  |         if (! (directiveName == QLatin1String("pragma") || | ||||||
|  |                directiveName == QLatin1String("import"))) | ||||||
|  |             return false; // not a valid directive name | ||||||
|  |  | ||||||
|  |         // it must be a pragma or an import directive. | ||||||
|  |         if (directiveName == QLatin1String("pragma")) { | ||||||
|  |             // .pragma library | ||||||
|  |             if (! (lex() == T_IDENTIFIER && tokenText() == QLatin1String("library"))) | ||||||
|  |                 return false; // expected `library | ||||||
|  |  | ||||||
|  |             // we found a .pragma library directive | ||||||
|  |             directives->pragmaLibrary(); | ||||||
|  |  | ||||||
|  |         } else { | ||||||
|  |             Q_ASSERT(directiveName == QLatin1String("import")); | ||||||
|  |             lex(); // skip .import | ||||||
|  |  | ||||||
|  |             QString pathOrUri; | ||||||
|  |             QString version; | ||||||
|  |             bool fileImport = false; // file or uri import | ||||||
|  |  | ||||||
|  |             if (_tokenKind == T_STRING_LITERAL) { | ||||||
|  |                 // .import T_STRING_LITERAL as T_IDENTIFIER | ||||||
|  |  | ||||||
|  |                 fileImport = true; | ||||||
|  |                 pathOrUri = tokenText(); | ||||||
|  |  | ||||||
|  |             } else if (_tokenKind == T_IDENTIFIER) { | ||||||
|  |                 // .import T_IDENTIFIER (. T_IDENTIFIER)* T_NUMERIC_LITERAL as T_IDENTIFIER | ||||||
|  |  | ||||||
|  |                 pathOrUri = tokenText(); | ||||||
|  |  | ||||||
|  |                 lex(); // skip the first T_IDENTIFIER | ||||||
|  |                 for (; _tokenKind == T_DOT; lex()) { | ||||||
|  |                     if (lex() != T_IDENTIFIER) | ||||||
|  |                         return false; | ||||||
|  |  | ||||||
|  |                     pathOrUri += QLatin1Char('.'); | ||||||
|  |                     pathOrUri += tokenText(); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 if (_tokenKind != T_NUMERIC_LITERAL) | ||||||
|  |                     return false; // expected the module version number | ||||||
|  |  | ||||||
|  |                 version = tokenText(); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // | ||||||
|  |             // recognize the mandatory `as' followed by the module name | ||||||
|  |             // | ||||||
|  |             if (! (lex() == T_RESERVED_WORD && tokenText() == QLatin1String("as"))) | ||||||
|  |                 return false; // expected `as' | ||||||
|  |  | ||||||
|  |             if (lex() != T_IDENTIFIER) | ||||||
|  |                 return false; // expected module name | ||||||
|  |  | ||||||
|  |             const QString module = tokenText(); | ||||||
|  |  | ||||||
|  |             if (fileImport) | ||||||
|  |                 directives->importFile(pathOrUri, module); | ||||||
|  |             else | ||||||
|  |                 directives->importModule(pathOrUri, version, module); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (tokenStartLine() != lineNumber) | ||||||
|  |             return false; // the directives cannot span over multiple lines | ||||||
|  |  | ||||||
|  |         // fetch the first token after the .pragma/.import directive | ||||||
|  |         lex(); | ||||||
|  |     } while (_tokenKind == T_DOT); | ||||||
|  |  | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
| #include "qmljskeywords_p.h" | #include "qmljskeywords_p.h" | ||||||
|   | |||||||
| @@ -54,6 +54,28 @@ namespace QmlJS { | |||||||
|  |  | ||||||
| class Engine; | class Engine; | ||||||
|  |  | ||||||
|  | class QML_PARSER_EXPORT Directives { | ||||||
|  | public: | ||||||
|  |     virtual ~Directives() {} | ||||||
|  |  | ||||||
|  |     virtual void pragmaLibrary() | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     virtual void importFile(const QString &jsfile, const QString &module) | ||||||
|  |     { | ||||||
|  |         Q_UNUSED(jsfile); | ||||||
|  |         Q_UNUSED(module); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     virtual void importModule(const QString &uri, const QString &version, const QString &module) | ||||||
|  |     { | ||||||
|  |         Q_UNUSED(uri); | ||||||
|  |         Q_UNUSED(version); | ||||||
|  |         Q_UNUSED(module); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  |  | ||||||
| class QML_PARSER_EXPORT Lexer: public QmlJSGrammar | class QML_PARSER_EXPORT Lexer: public QmlJSGrammar | ||||||
| { | { | ||||||
| public: | public: | ||||||
| @@ -105,6 +127,12 @@ public: | |||||||
|         EqualPrefix |         EqualPrefix | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|  |     enum RegExpFlag { | ||||||
|  |         RegExp_Global     = 0x01, | ||||||
|  |         RegExp_IgnoreCase = 0x02, | ||||||
|  |         RegExp_Multiline  = 0x04 | ||||||
|  |     }; | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     Lexer(Engine *engine); |     Lexer(Engine *engine); | ||||||
|  |  | ||||||
| @@ -114,10 +142,12 @@ public: | |||||||
|     int lex(); |     int lex(); | ||||||
|  |  | ||||||
|     bool scanRegExp(RegExpBodyPrefix prefix = NoPrefix); |     bool scanRegExp(RegExpBodyPrefix prefix = NoPrefix); | ||||||
|  |     bool scanDirectives(Directives *directives); | ||||||
|  |  | ||||||
|     int regExpFlags() const { return _patternFlags; } |     int regExpFlags() const { return _patternFlags; } | ||||||
|     QString regExpPattern() const { return _tokenText; } |     QString regExpPattern() const { return _tokenText; } | ||||||
|  |  | ||||||
|  |     int tokenKind() const; | ||||||
|     int tokenOffset() const; |     int tokenOffset() const; | ||||||
|     int tokenLength() const; |     int tokenLength() const; | ||||||
|  |  | ||||||
| @@ -135,6 +165,8 @@ public: | |||||||
|     QString errorMessage() const; |     QString errorMessage() const; | ||||||
|  |  | ||||||
|     bool prevTerminator() const; |     bool prevTerminator() const; | ||||||
|  |     bool followsClosingBrace() const; | ||||||
|  |     bool canInsertAutomaticSemicolon(int token) const; | ||||||
|  |  | ||||||
|     enum ParenthesesState { |     enum ParenthesesState { | ||||||
|         IgnoreParentheses, |         IgnoreParentheses, | ||||||
| @@ -192,6 +224,7 @@ private: | |||||||
|     bool _prohibitAutomaticSemicolon; |     bool _prohibitAutomaticSemicolon; | ||||||
|     bool _restrictedKeyword; |     bool _restrictedKeyword; | ||||||
|     bool _terminator; |     bool _terminator; | ||||||
|  |     bool _followsClosingBrace; | ||||||
|     bool _delimited; |     bool _delimited; | ||||||
|     bool _qmlMode; |     bool _qmlMode; | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ | |||||||
| **************************************************************************/ | **************************************************************************/ | ||||||
|  |  | ||||||
| #include <QtCore/QtDebug> | #include <QtCore/QtDebug> | ||||||
| #include <QtGui/QApplication> | #include <QtCore/QCoreApplication> | ||||||
|  |  | ||||||
| #include <string.h> | #include <string.h> | ||||||
|  |  | ||||||
| @@ -67,14 +67,6 @@ void Parser::reallocateStack() | |||||||
|     string_stack = reinterpret_cast<QStringRef*> (qRealloc(string_stack, stack_size * sizeof(QStringRef))); |     string_stack = reinterpret_cast<QStringRef*> (qRealloc(string_stack, stack_size * sizeof(QStringRef))); | ||||||
| } | } | ||||||
|  |  | ||||||
| inline static bool automatic(Engine *driver, int token) |  | ||||||
| { |  | ||||||
|     return token == QmlJSGrammar::T_RBRACE |  | ||||||
|         || token == 0 |  | ||||||
|         || driver->lexer()->prevTerminator(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Parser::Parser(Engine *engine): | Parser::Parser(Engine *engine): | ||||||
|     driver(engine), |     driver(engine), | ||||||
|     pool(engine->pool()), |     pool(engine->pool()), | ||||||
| @@ -1687,7 +1679,7 @@ case 342: { | |||||||
|         const int errorState = state_stack[tos]; |         const int errorState = state_stack[tos]; | ||||||
|  |  | ||||||
|         // automatic insertion of `;' |         // automatic insertion of `;' | ||||||
|         if (yytoken != -1 && t_action(errorState, T_AUTOMATIC_SEMICOLON) && automatic(driver, yytoken)) { |         if (yytoken != -1 && t_action(errorState, T_AUTOMATIC_SEMICOLON) && lexer->canInsertAutomaticSemicolon(yytoken)) { | ||||||
|             SavedToken &tk = token_buffer[0]; |             SavedToken &tk = token_buffer[0]; | ||||||
|             tk.token = yytoken; |             tk.token = yytoken; | ||||||
|             tk.dval = yylval; |             tk.dval = yylval; | ||||||
|   | |||||||
| @@ -1,82 +0,0 @@ | |||||||
| /************************************************************************** |  | ||||||
| ** |  | ||||||
| ** This file is part of Qt Creator |  | ||||||
| ** |  | ||||||
| ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). |  | ||||||
| ** |  | ||||||
| ** Contact: Nokia Corporation (info@qt.nokia.com) |  | ||||||
| ** |  | ||||||
| ** |  | ||||||
| ** GNU Lesser General Public License Usage |  | ||||||
| ** |  | ||||||
| ** This file may be used under the terms of the GNU Lesser General Public |  | ||||||
| ** License version 2.1 as published by the Free Software Foundation and |  | ||||||
| ** appearing in the file LICENSE.LGPL included in the packaging of this file. |  | ||||||
| ** Please review the following information to ensure the GNU Lesser General |  | ||||||
| ** Public License version 2.1 requirements will be met: |  | ||||||
| ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. |  | ||||||
| ** |  | ||||||
| ** In addition, as a special exception, Nokia gives you certain additional |  | ||||||
| ** rights. These rights are described in the Nokia Qt LGPL Exception |  | ||||||
| ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. |  | ||||||
| ** |  | ||||||
| ** Other Usage |  | ||||||
| ** |  | ||||||
| ** Alternatively, this file may be used in accordance with the terms and |  | ||||||
| ** conditions contained in a signed written agreement between you and Nokia. |  | ||||||
| ** |  | ||||||
| ** If you have questions regarding the use of this file, please contact |  | ||||||
| ** Nokia at info@qt.nokia.com. |  | ||||||
| ** |  | ||||||
| **************************************************************************/ |  | ||||||
|  |  | ||||||
| #ifndef QMLUTIL_P_H |  | ||||||
| #define QMLUTIL_P_H |  | ||||||
|  |  | ||||||
| #include <QtCore/QString> |  | ||||||
|  |  | ||||||
| QT_BEGIN_NAMESPACE |  | ||||||
|  |  | ||||||
| namespace QmlUtils { |  | ||||||
|  |  | ||||||
| inline bool isUpper(const QChar &qc) |  | ||||||
| { |  | ||||||
|     ushort c = qc.unicode(); |  | ||||||
|     return ((c >= 'A' && c <= 'Z') || (c > 127 && QChar::category(c) == QChar::Letter_Uppercase)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline bool isLower(const QChar &qc) |  | ||||||
| { |  | ||||||
|     ushort c = qc.unicode(); |  | ||||||
|     return ((c >= 'a' && c <= 'z') || (c > 127 && QChar::category(c) == QChar::Letter_Lowercase)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline bool isLetter(const QChar &qc) |  | ||||||
| { |  | ||||||
|     ushort c = qc.unicode(); |  | ||||||
|     return ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c > 127 && qc.isLetter())); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline bool isDigit(const QChar &qc) |  | ||||||
| { |  | ||||||
|     ushort c = qc.unicode(); |  | ||||||
|     return ((c >= '0' && c <= '9') || (c > 127 && qc.isDigit())); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline bool isLetterOrNumber(const QChar &qc) |  | ||||||
| { |  | ||||||
|     ushort c = qc.unicode(); |  | ||||||
|     return ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9') || (c > 127 && qc.isLetterOrNumber())); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline bool isSpace(const QChar &qc) |  | ||||||
| { |  | ||||||
|     ushort c = qc.unicode(); |  | ||||||
|     return (c == 0x20 || (c >= 0x09 && c <= 0x0D) || c == 0x85 || (c > 127 && qc.isSpace())); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| } // namespace Qml |  | ||||||
|  |  | ||||||
| QT_END_NAMESPACE |  | ||||||
|  |  | ||||||
| #endif // QMLUTIL_P_H |  | ||||||
		Reference in New Issue
	
	Block a user