com.pinterest.ktlint.ruleset.standard.rules.FunctionStartOfBodySpacingRule.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
import com.pinterest.ktlint.rule.engine.core.api.ElementType.FUN
import com.pinterest.ktlint.rule.engine.core.api.RuleId
import com.pinterest.ktlint.rule.engine.core.api.nextLeaf
import com.pinterest.ktlint.rule.engine.core.api.prevLeaf
import com.pinterest.ktlint.rule.engine.core.api.upsertWhitespaceAfterMe
import com.pinterest.ktlint.rule.engine.core.api.upsertWhitespaceBeforeMe
import com.pinterest.ktlint.ruleset.standard.StandardRule
import org.jetbrains.kotlin.com.intellij.lang.ASTNode
/**
* Lints and formats the spacing after the fun keyword
*/
public class FunctionStartOfBodySpacingRule : StandardRule("function-start-of-body-spacing") {
override fun beforeVisitChildNodes(
node: ASTNode,
autoCorrect: Boolean,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
) {
if (node.elementType == FUN) {
node
.findChildByType(ElementType.EQ)
?.let { visitFunctionFollowedByBodyExpression(node, emit, autoCorrect) }
node
.findChildByType(ElementType.BLOCK)
?.let { visitFunctionFollowedByBodyBlock(node, emit, autoCorrect) }
}
}
private fun visitFunctionFollowedByBodyExpression(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
autoCorrect: Boolean,
) {
fixWhiteSpaceBeforeAssignmentOfBodyExpression(node, emit, autoCorrect)
fixWhiteSpaceBetweenAssignmentAndBodyExpression(node, emit, autoCorrect)
}
private fun fixWhiteSpaceBeforeAssignmentOfBodyExpression(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
autoCorrect: Boolean,
) {
node
.findChildByType(ElementType.EQ)
?.let { assignmentExpression ->
assignmentExpression
.prevLeaf(includeEmpty = true)
?.takeIf { it.elementType == ElementType.WHITE_SPACE }
.let { whiteSpaceBeforeAssignment ->
if (whiteSpaceBeforeAssignment == null) {
emit(
assignmentExpression.startOffset,
"Expected a single white space before assignment of expression body",
true,
)
if (autoCorrect) {
assignmentExpression.upsertWhitespaceBeforeMe(" ")
}
} else if (whiteSpaceBeforeAssignment.text != " ") {
emit(whiteSpaceBeforeAssignment.startOffset, "Unexpected whitespace", true)
if (autoCorrect) {
assignmentExpression.upsertWhitespaceBeforeMe(" ")
}
}
}
}
}
private fun fixWhiteSpaceBetweenAssignmentAndBodyExpression(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
autoCorrect: Boolean,
) {
node
.findChildByType(ElementType.EQ)
?.let { assignmentExpression ->
assignmentExpression
.nextLeaf(includeEmpty = true)
?.takeIf { it.elementType == ElementType.WHITE_SPACE }
.let { whiteSpaceAfterAssignment ->
if (!(whiteSpaceAfterAssignment?.text == " " || whiteSpaceAfterAssignment?.textContains('\n') == true)) {
emit(
assignmentExpression.startOffset,
"Expected a single white space between assignment and expression body on same line",
true,
)
if (autoCorrect) {
assignmentExpression.upsertWhitespaceAfterMe(" ")
}
}
}
}
}
private fun visitFunctionFollowedByBodyBlock(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> Unit,
autoCorrect: Boolean,
) {
node
.findChildByType(ElementType.BLOCK)
?.let { block ->
block
.prevLeaf(includeEmpty = true)
?.takeIf { it.elementType == ElementType.WHITE_SPACE }
.let { whiteSpaceBeforeExpressionBlock ->
if (whiteSpaceBeforeExpressionBlock?.text != " ") {
emit(block.startOffset, "Expected a single white space before start of function body", true)
if (autoCorrect) {
block
.firstChildNode
.prevLeaf(true)
?.upsertWhitespaceAfterMe(" ")
}
}
}
}
}
}
public val FUNCTION_START_OF_BODY_SPACING_RULE_ID: RuleId = FunctionStartOfBodySpacingRule().ruleId
© 2015 - 2024 Weber Informatics LLC | Privacy Policy