com.pinterest.ktlint.ruleset.standard.rules.IfElseWrappingRule.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ktlint-ruleset-standard Show documentation
Show all versions of ktlint-ruleset-standard Show documentation
An anti-bikeshedding Kotlin linter with built-in formatter.
package com.pinterest.ktlint.ruleset.standard.rules
import com.pinterest.ktlint.rule.engine.core.api.ElementType.BLOCK
import com.pinterest.ktlint.rule.engine.core.api.ElementType.ELSE
import com.pinterest.ktlint.rule.engine.core.api.ElementType.ELSE_KEYWORD
import com.pinterest.ktlint.rule.engine.core.api.ElementType.IF
import com.pinterest.ktlint.rule.engine.core.api.ElementType.IF_KEYWORD
import com.pinterest.ktlint.rule.engine.core.api.ElementType.LBRACE
import com.pinterest.ktlint.rule.engine.core.api.ElementType.RBRACE
import com.pinterest.ktlint.rule.engine.core.api.ElementType.THEN
import com.pinterest.ktlint.rule.engine.core.api.IndentConfig
import com.pinterest.ktlint.rule.engine.core.api.Rule
import com.pinterest.ktlint.rule.engine.core.api.Rule.VisitorModifier.RunAfterRule
import com.pinterest.ktlint.rule.engine.core.api.Rule.VisitorModifier.RunAfterRule.Mode.ONLY_WHEN_RUN_AFTER_RULE_IS_LOADED_AND_ENABLED
import com.pinterest.ktlint.rule.engine.core.api.RuleId
import com.pinterest.ktlint.rule.engine.core.api.SinceKtlint
import com.pinterest.ktlint.rule.engine.core.api.SinceKtlint.Status.EXPERIMENTAL
import com.pinterest.ktlint.rule.engine.core.api.SinceKtlint.Status.STABLE
import com.pinterest.ktlint.rule.engine.core.api.children
import com.pinterest.ktlint.rule.engine.core.api.editorconfig.EditorConfig
import com.pinterest.ktlint.rule.engine.core.api.editorconfig.INDENT_SIZE_PROPERTY
import com.pinterest.ktlint.rule.engine.core.api.editorconfig.INDENT_STYLE_PROPERTY
import com.pinterest.ktlint.rule.engine.core.api.indent
import com.pinterest.ktlint.rule.engine.core.api.isPartOfComment
import com.pinterest.ktlint.rule.engine.core.api.isWhiteSpaceWithNewline
import com.pinterest.ktlint.rule.engine.core.api.isWhiteSpaceWithoutNewline
import com.pinterest.ktlint.rule.engine.core.api.nextCodeLeaf
import com.pinterest.ktlint.rule.engine.core.api.nextLeaf
import com.pinterest.ktlint.rule.engine.core.api.nextSibling
import com.pinterest.ktlint.rule.engine.core.api.prevCodeLeaf
import com.pinterest.ktlint.rule.engine.core.api.prevLeaf
import com.pinterest.ktlint.rule.engine.core.api.upsertWhitespaceBeforeMe
import com.pinterest.ktlint.ruleset.standard.StandardRule
import org.jetbrains.kotlin.com.intellij.lang.ASTNode
import org.jetbrains.kotlin.psi.psiUtil.parents
import org.jetbrains.kotlin.utils.addToStdlib.applyIf
/**
* Enforce that single line if statements are kept simple. A single line if statement is allowed only when it has at most one else branch.
* Also, the branches of such an if statement may not be wrapped in a block.
*/
@SinceKtlint("0.49", EXPERIMENTAL)
@SinceKtlint("1.0", STABLE)
public class IfElseWrappingRule :
StandardRule(
id = "if-else-wrapping",
visitorModifiers = setOf(RunAfterRule(DISCOURAGED_COMMENT_LOCATION_RULE_ID, ONLY_WHEN_RUN_AFTER_RULE_IS_LOADED_AND_ENABLED)),
usesEditorConfigProperties =
setOf(
INDENT_SIZE_PROPERTY,
INDENT_STYLE_PROPERTY,
),
),
Rule.OfficialCodeStyle {
private var indentConfig = IndentConfig.DEFAULT_INDENT_CONFIG
override fun beforeFirstNode(editorConfig: EditorConfig) {
indentConfig =
IndentConfig(
indentStyle = editorConfig[INDENT_STYLE_PROPERTY],
tabWidth = editorConfig[INDENT_SIZE_PROPERTY],
)
}
override fun beforeVisitChildNodes(
node: ASTNode,
autoCorrect: Boolean,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
) {
if (node.elementType == IF) {
visitIf(node, autoCorrect, emit)
}
}
private fun visitIf(
node: ASTNode,
autoCorrect: Boolean,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
) {
val outerIf = node.outerIf()
val multilineIf = outerIf.textContains('\n')
val nestedIf = outerIf.isNestedIf()
with(node) {
findChildByType(THEN)
?.let { visitElement(it, autoCorrect, emit, multilineIf, nestedIf) }
findChildByType(ELSE_KEYWORD)
?.let { visitElement(it, autoCorrect, emit, multilineIf, nestedIf) }
findChildByType(ELSE)
?.let { visitElement(it, autoCorrect, emit, multilineIf, nestedIf) }
}
}
private fun visitElement(
node: ASTNode,
autoCorrect: Boolean,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
multilineIf: Boolean,
nestedIf: Boolean,
) {
if (!multilineIf) {
visitBranchSingleLineIf(node, emit)
}
if (multilineIf || nestedIf) {
visitBranch(node, autoCorrect, emit, multilineIf)
}
}
private fun visitBranchSingleLineIf(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
) {
node
.findChildByType(BLOCK)
?.let {
// The then or else block is part of single line if-statement. To enforce that such statements are readable and
// simple, blocks are forbidden entirely in single line if-statements.
emit(
node.startOffset,
"A single line if-statement should be kept simple. The '${node.elementType}' may not be wrapped in a block.",
false,
)
}
}
private fun visitBranch(
node: ASTNode,
autoCorrect: Boolean,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
multilineIf: Boolean,
) {
if (multilineIf) {
if (node.elementType == IF && node.prevCodeLeaf()?.elementType == ELSE_KEYWORD ||
node.elementType == ELSE && node.nextCodeLeaf()?.elementType == IF_KEYWORD ||
node.elementType == ELSE_KEYWORD && node.nextCodeLeaf()?.elementType == IF_KEYWORD
) {
// Allow "else if" on single line
return
}
if (node.elementType == ELSE_KEYWORD && node.nextCodeLeaf()?.elementType == LBRACE) {
// Allow "else {" on single line
return
}
} else {
// Outer if statement is a single line statement
if (node.elementType == ELSE && node.nextCodeLeaf()?.elementType == IF_KEYWORD) {
// Ignore "else if" as it is reported via another message
return
}
}
with(node.findFirstNodeInBlockToBeIndented() ?: node) {
val expectedIndent =
if (nextSibling()?.elementType == RBRACE) {
node.indent()
} else {
indentConfig.siblingIndentOf(node)
}
applyIf(elementType == THEN || elementType == ELSE || elementType == ELSE_KEYWORD) { prevLeaf()!! }
.takeUnless { it.isWhiteSpaceWithNewline() }
?.let {
// Expected a newline with indent. Leave it up to the IndentationRule to determine exact indent
emit(startOffset, "Expected a newline", true)
if (autoCorrect) {
upsertWhitespaceBeforeMe(expectedIndent)
}
}
}
}
private fun ASTNode.findFirstNodeInBlockToBeIndented(): ASTNode? =
findChildByType(BLOCK)
?.children()
?.first {
it.elementType != LBRACE &&
!it.isWhitespaceBeforeComment() &&
!it.isPartOfComment()
}
private fun ASTNode.isWhitespaceBeforeComment() = isWhiteSpaceWithoutNewline() && nextLeaf()?.isPartOfComment() == true
private fun ASTNode.outerIf(): ASTNode {
require(this.elementType == IF)
return parents()
.takeWhile { it.elementType in IF_THEN_ELSE_ELEMENT_TYPES }
.lastOrNull()
?: this
}
private fun ASTNode.isNestedIf(): Boolean {
require(elementType == IF)
return findChildByType(THEN)?.firstChildNode?.elementType == IF ||
findChildByType(ELSE)?.firstChildNode?.elementType == IF
}
private companion object {
val IF_THEN_ELSE_ELEMENT_TYPES =
listOf(
IF,
THEN,
ELSE,
)
}
}
public val IF_ELSE_WRAPPING_RULE_ID: RuleId = IfElseWrappingRule().ruleId
© 2015 - 2024 Weber Informatics LLC | Privacy Policy