com.pinterest.ktlint.ruleset.standard.rules.SpacingAroundColonRule.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.AutocorrectDecision
import com.pinterest.ktlint.rule.engine.core.api.ElementType.COLON
import com.pinterest.ktlint.rule.engine.core.api.ElementType.EQ
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.STABLE
import com.pinterest.ktlint.rule.engine.core.api.ifAutocorrectAllowed
import com.pinterest.ktlint.rule.engine.core.api.isPartOfComment
import com.pinterest.ktlint.rule.engine.core.api.isWhiteSpace
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.nextLeaf
import com.pinterest.ktlint.rule.engine.core.api.nextSibling
import com.pinterest.ktlint.rule.engine.core.api.prevLeaf
import com.pinterest.ktlint.rule.engine.core.api.prevSibling
import com.pinterest.ktlint.rule.engine.core.api.remove
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
import org.jetbrains.kotlin.com.intellij.psi.impl.source.tree.LeafPsiElement
import org.jetbrains.kotlin.psi.KtAnnotation
import org.jetbrains.kotlin.psi.KtAnnotationEntry
import org.jetbrains.kotlin.psi.KtBlockExpression
import org.jetbrains.kotlin.psi.KtClassOrObject
import org.jetbrains.kotlin.psi.KtConstructor
import org.jetbrains.kotlin.psi.KtNamedFunction
import org.jetbrains.kotlin.psi.KtProperty
import org.jetbrains.kotlin.psi.KtTypeConstraint
import org.jetbrains.kotlin.psi.KtTypeParameterList
import org.jetbrains.kotlin.psi.psiUtil.siblings
import org.jetbrains.kotlin.utils.addToStdlib.firstIsInstanceOrNull
@SinceKtlint("0.1", STABLE)
public class SpacingAroundColonRule : StandardRule("colon-spacing") {
override fun beforeVisitChildNodes(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> AutocorrectDecision,
) {
if (node.elementType == COLON) {
removeUnexpectedNewlineBefore(node, emit)
removeUnexpectedSpacingAround(node, emit)
addMissingSpacingAround(node, emit)
}
}
private fun removeUnexpectedNewlineBefore(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> AutocorrectDecision,
) {
val psiParent = node.psi.parent
val prevLeaf = node.prevLeaf()
if (prevLeaf != null && prevLeaf.isWhiteSpaceWithNewline()) {
emit(prevLeaf.startOffset, "Unexpected newline before \":\"", true)
.ifAutocorrectAllowed {
val prevNonCodeElements =
node
.siblings(forward = false)
.takeWhile { it.isWhiteSpace() || it.isPartOfComment() }
.toList()
.reversed()
when {
psiParent is KtProperty || psiParent is KtNamedFunction -> {
val equalsSignElement =
node
.siblings(forward = true)
.firstOrNull { it.elementType == EQ }
if (equalsSignElement != null) {
equalsSignElement
.treeNext
?.let { treeNext ->
prevNonCodeElements.forEach {
node.treeParent.addChild(it, treeNext)
}
if (treeNext.isWhiteSpace()) {
treeNext.remove()
}
Unit
}
}
val blockElement =
node
.siblings(forward = true)
.firstIsInstanceOrNull()
if (blockElement != null) {
val before =
blockElement
.firstChildNode
.nextSibling()
prevNonCodeElements
.let {
if (it.first().isWhiteSpace()) {
it.first().remove()
it.drop(1)
}
if (it.last().isWhiteSpaceWithNewline()) {
it.last().remove()
it.dropLast(1)
} else {
it
}
}.forEach {
blockElement.addChild(it, before)
}
}
}
prevLeaf.prevLeaf()?.isPartOfComment() == true -> {
val nextLeaf = node.nextLeaf()
prevNonCodeElements.forEach {
node.treeParent.addChild(it, nextLeaf)
}
if (nextLeaf != null && nextLeaf.isWhiteSpace()) {
nextLeaf.remove()
}
}
else -> {
val text = prevLeaf.text
if (node.spacingBefore) {
(prevLeaf as LeafPsiElement).rawReplaceWithText(" ")
} else {
prevLeaf.remove()
}
node.upsertWhitespaceAfterMe(text)
}
}
}
}
}
private fun removeUnexpectedSpacingAround(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> AutocorrectDecision,
) {
if (node.prevSibling().isWhiteSpaceWithoutNewline() && node.noSpacingBefore) {
emit(node.startOffset, "Unexpected spacing before \":\"", true)
.ifAutocorrectAllowed {
node
.prevSibling()
?.remove()
}
}
if (node.nextSibling().isWhiteSpaceWithoutNewline() && node.spacingAfter) {
emit(node.startOffset, "Unexpected spacing after \":\"", true)
.ifAutocorrectAllowed {
node
.nextSibling()
?.remove()
}
}
}
private fun addMissingSpacingAround(
node: ASTNode,
emit: (offset: Int, errorMessage: String, canBeAutoCorrected: Boolean) -> AutocorrectDecision,
) {
val missingSpacingBefore = !node.prevSibling().isWhiteSpace() && node.spacingBefore
val missingSpacingAfter = !node.nextSibling().isWhiteSpace() && node.noSpacingAfter
when {
missingSpacingBefore && missingSpacingAfter -> {
emit(node.startOffset, "Missing spacing around \":\"", true)
.ifAutocorrectAllowed {
node.upsertWhitespaceBeforeMe(" ")
node.upsertWhitespaceAfterMe(" ")
}
}
missingSpacingBefore -> {
emit(node.startOffset, "Missing spacing before \":\"", true)
.ifAutocorrectAllowed {
node.upsertWhitespaceBeforeMe(" ")
}
}
missingSpacingAfter -> {
emit(node.startOffset + 1, "Missing spacing after \":\"", true)
.ifAutocorrectAllowed {
node.upsertWhitespaceAfterMe(" ")
}
}
}
}
private inline val ASTNode.spacingBefore: Boolean
get() =
when {
psi.parent is KtClassOrObject -> {
true
}
psi.parent is KtConstructor<*> -> {
// constructor : this/super
true
}
psi.parent is KtTypeConstraint -> {
// where T : S
true
}
psi.parent.parent is KtTypeParameterList -> {
true
}
else -> {
false
}
}
private inline val ASTNode.noSpacingBefore: Boolean
get() = !spacingBefore
private inline val ASTNode.spacingAfter: Boolean
get() =
when (psi.parent) {
is KtAnnotation, is KtAnnotationEntry -> true
else -> false
}
private inline val ASTNode.noSpacingAfter: Boolean
get() = !spacingAfter
}
public val SPACING_AROUND_COLON_RULE_ID: RuleId = SpacingAroundColonRule().ruleId
© 2015 - 2024 Weber Informatics LLC | Privacy Policy