
com.github.shyiko.ktlint.ruleset.standard.SpacingAroundKeywordRule.kt Maven / Gradle / Ivy
The newest version!
package com.github.shyiko.ktlint.ruleset.standard
import com.github.shyiko.ktlint.core.Rule
import com.github.shyiko.ktlint.core.ast.ElementType.CATCH_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.DO_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.ELSE_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.FINALLY_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.FOR_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.GET_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.IF_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.RBRACE
import com.github.shyiko.ktlint.core.ast.ElementType.SET_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.TRY_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.WHEN_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.WHILE_KEYWORD
import com.github.shyiko.ktlint.core.ast.ElementType.WHITE_SPACE
import com.github.shyiko.ktlint.core.ast.nextLeaf
import com.github.shyiko.ktlint.core.ast.prevLeaf
import com.github.shyiko.ktlint.core.ast.upsertWhitespaceAfterMe
import org.jetbrains.kotlin.com.intellij.lang.ASTNode
import org.jetbrains.kotlin.com.intellij.psi.PsiWhiteSpace
import org.jetbrains.kotlin.com.intellij.psi.impl.source.tree.LeafElement
import org.jetbrains.kotlin.com.intellij.psi.impl.source.tree.LeafPsiElement
import org.jetbrains.kotlin.com.intellij.psi.tree.TokenSet.create
import org.jetbrains.kotlin.psi.KtPropertyAccessor
import org.jetbrains.kotlin.psi.KtWhenEntry
class SpacingAroundKeywordRule : Rule("keyword-spacing") {
private val noLFBeforeSet = create(ELSE_KEYWORD, CATCH_KEYWORD, FINALLY_KEYWORD)
private val tokenSet = create(
FOR_KEYWORD, IF_KEYWORD, ELSE_KEYWORD, WHILE_KEYWORD, DO_KEYWORD,
TRY_KEYWORD, CATCH_KEYWORD, FINALLY_KEYWORD, WHEN_KEYWORD
)
private val keywordsWithoutSpaces = create(GET_KEYWORD, SET_KEYWORD)
override fun visit(
node: ASTNode,
autoCorrect: Boolean,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit
) {
if (node is LeafPsiElement) {
if (tokenSet.contains(node.elementType) && node.nextLeaf() !is PsiWhiteSpace) {
emit(node.startOffset + node.text.length, "Missing spacing after \"${node.text}\"", true)
if (autoCorrect) {
node.upsertWhitespaceAfterMe(" ")
}
} else if (keywordsWithoutSpaces.contains(node.elementType) && node.nextLeaf() is PsiWhiteSpace) {
val parent = node.parent
val nextLeaf = node.nextLeaf()
if (parent is KtPropertyAccessor && parent.hasBody() && nextLeaf != null) {
emit(node.startOffset, "Unexpected spacing after \"${node.text}\"", true)
if (autoCorrect) {
nextLeaf.treeParent.removeChild(nextLeaf)
}
}
}
if (noLFBeforeSet.contains(node.elementType)) {
val prevLeaf = node.prevLeaf()
if (
prevLeaf?.elementType == WHITE_SPACE &&
prevLeaf.textContains('\n') &&
(node.elementType != ELSE_KEYWORD || node.parent !is KtWhenEntry)
) {
val presumablyCurly = prevLeaf.prevLeaf()
if (presumablyCurly != null &&
presumablyCurly.elementType == RBRACE &&
(
node.elementType != ELSE_KEYWORD ||
// `if (...) v.let { } else` case
presumablyCurly.treeParent?.treeParent?.treeParent == node.treeParent
)
) {
emit(node.startOffset, "Unexpected newline before \"${node.text}\"", true)
if (autoCorrect) {
(prevLeaf as LeafElement).rawReplaceWithText(" ")
}
}
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy