blob: 274267382e3f8b9406ceb4475b05a6b2d3a2dc73 [file] [log] [blame]
/*
* Copyright 2000-2009 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.intellij.lang;
import com.intellij.lexer.Lexer;
import com.intellij.openapi.project.Project;
import com.intellij.psi.FileViewProvider;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiFile;
import com.intellij.psi.tree.IFileElementType;
import com.intellij.psi.tree.TokenSet;
import org.jetbrains.annotations.NotNull;
/**
* Defines the implementation of a parser for a custom language.
*
* @see LanguageParserDefinitions#forLanguage(Language)
*/
public interface ParserDefinition {
/**
* Returns the lexer for lexing files in the specified project. This lexer does not need to support incremental relexing - it is always
* called for the entire file.
*
* @param project the project to which the lexer is connected.
* @return the lexer instance.
*/
@NotNull
Lexer createLexer(Project project);
/**
* Returns the parser for parsing files in the specified project.
*
* @param project the project to which the parser is connected.
* @return the parser instance.
*/
PsiParser createParser(Project project);
/**
* Returns the element type of the node describing a file in the specified language.
*
* @return the file node element type.
*/
IFileElementType getFileNodeType();
/**
* Returns the set of token types which are treated as whitespace by the PSI builder.
* Tokens of those types are automatically skipped by PsiBuilder. Whitespace elements
* on the bounds of nodes built by PsiBuilder are automatically excluded from the text
* range of the nodes.
* <p><strong>It is strongly advised you return TokenSet that only contains {@link com.intellij.psi.TokenType#WHITE_SPACE},
* which is suitable for all the languages unless you really need to use special whitespace token</strong>
*
* @return the set of whitespace token types.
*/
@NotNull
TokenSet getWhitespaceTokens();
/**
* Returns the set of token types which are treated as comments by the PSI builder.
* Tokens of those types are automatically skipped by PsiBuilder. Also, To Do patterns
* are searched in the text of tokens of those types.
*
* @return the set of comment token types.
*/
@NotNull
TokenSet getCommentTokens();
/**
* Returns the set of element types which are treated as string literals. "Search in strings"
* option in refactorings is applied to the contents of such tokens.
*
* @return the set of string literal element types.
*/
@NotNull
TokenSet getStringLiteralElements();
/**
* Creates a PSI element for the specified AST node. The AST tree is a simple, semantic-free
* tree of AST nodes which is built during the PsiBuilder parsing pass. The PSI tree is built
* over the AST tree and includes elements of different types for different language constructs.
*
* !!!WARNING!!! PSI element types should be unambiguously determined by AST node element types.
* You can not produce different PSI elements from AST nodes of the same types (e.g. based on AST node content).
* Typically, your code should be as simple as that:
* <code>
* if (node.getElementType == MY_ELEMENT_TYPE) {
* return new MyPsiElement(node);
* }
* </code>
*
* @param node the node for which the PSI element should be returned.
* @return the PSI element matching the element type of the AST node.
*/
@NotNull
PsiElement createElement(ASTNode node);
/**
* Creates a PSI element for the specified virtual file.
*
* @param viewProvider virtual file.
* @return the PSI file element.
*/
PsiFile createFile(FileViewProvider viewProvider);
/**
* Checks if the specified two token types need to be separated by a space according to the language grammar.
* For example, in Java two keywords are always separated by a space; a keyword and an opening parenthesis may
* be separated or not separated. This is used for automatic whitespace insertion during AST modification operations.
*
* @param left the first token to check.
* @param right the second token to check.
* @return the spacing requirements.
* @since 6.0
*/
SpaceRequirements spaceExistanceTypeBetweenTokens(ASTNode left, ASTNode right);
/**
* Requirements for spacing between tokens.
*
* @see ParserDefinition#spaceExistanceTypeBetweenTokens
*/
enum SpaceRequirements {
/** Whitespace between tokens is optional. */
MAY,
/** Whitespace between tokens is required. */
MUST,
/** Whitespace between tokens is not allowed. */
MUST_NOT,
/** A line break is required between tokens. */
MUST_LINE_BREAK,
}
}