summaryrefslogtreecommitdiff
path: root/plugins/kotlin/frontend-independent/src/org/jetbrains/kotlin/idea/quickfix/AddElseBranchFix.kt
blob: addcad5b933d66a8eae88788a1e26b497421c472 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
// Copyright 2000-2022 JetBrains s.r.o. and contributors. Use of this source code is governed by the Apache 2.0 license.

package org.jetbrains.kotlin.idea.quickfix

import com.intellij.codeInsight.CodeInsightUtilCore
import com.intellij.codeInsight.intention.IntentionAction
import com.intellij.openapi.editor.Editor
import com.intellij.openapi.project.Project
import com.intellij.psi.PsiDocumentManager
import com.intellij.psi.PsiElement
import com.intellij.psi.PsiErrorElement
import com.intellij.psi.util.elementType
import org.jetbrains.kotlin.KtNodeTypes
import org.jetbrains.kotlin.idea.KotlinBundle
import org.jetbrains.kotlin.idea.util.executeEnterHandler
import org.jetbrains.kotlin.psi.*
import org.jetbrains.kotlin.psi.psiUtil.endOffset
import org.jetbrains.kotlin.psi.psiUtil.getNonStrictParentOfType
import org.jetbrains.kotlin.utils.addToStdlib.safeAs

sealed class AddElseBranchFix<T : KtExpression>(element: T) : KotlinPsiOnlyQuickFixAction<T>(element) {
    override fun getFamilyName() = KotlinBundle.message("fix.add.else.branch.when")
    override fun getText() = familyName

    abstract override fun isAvailable(project: Project, editor: Editor?, file: KtFile): Boolean

    abstract override fun invoke(project: Project, editor: Editor?, file: KtFile)
}

class AddWhenElseBranchFix(element: KtWhenExpression) : AddElseBranchFix<KtWhenExpression>(element) {
    override fun isAvailable(project: Project, editor: Editor?, file: KtFile): Boolean = element?.closeBrace != null

    override fun invoke(project: Project, editor: Editor?, file: KtFile) {
        val element = element ?: return
        val whenCloseBrace = element.closeBrace ?: return
        val entry = KtPsiFactory(file).createWhenEntry("else -> {}")
        CodeInsightUtilCore.forcePsiPostprocessAndRestoreElement(element.addBefore(entry, whenCloseBrace))?.endOffset?.let { offset ->
            editor?.caretModel?.moveToOffset(offset - 1)
        }
    }

    companion object : QuickFixesPsiBasedFactory<PsiElement>(PsiElement::class, PsiElementSuitabilityCheckers.ALWAYS_SUITABLE) {
        override fun doCreateQuickFix(psiElement: PsiElement): List<IntentionAction> {
            return listOfNotNull(psiElement.getNonStrictParentOfType<KtWhenExpression>()?.let(::AddWhenElseBranchFix))
        }
    }
}

class AddIfElseBranchFix(element: KtIfExpression) : AddElseBranchFix<KtIfExpression>(element) {
    override fun isAvailable(project: Project, editor: Editor?, file: KtFile): Boolean {
        val ktIfExpression = element ?: return false
        return ktIfExpression.`else` == null &&
                ktIfExpression.condition != null &&
                ktIfExpression.children.firstOrNull { it.elementType == KtNodeTypes.THEN }?.firstChild !is PsiErrorElement
    }

    override fun invoke(project: Project, editor: Editor?, file: KtFile) {
        val element = element ?: return
        val withBraces = element.then is KtBlockExpression
        val psiFactory = KtPsiFactory(file)
        val newIf = psiFactory.createExpression(
            if (withBraces) {
                "if (true) {} else {}"
            } else {
                "if (true) 2 else TODO()"
            }
        ) as KtIfExpression

        element.addRange(newIf.then?.parent?.nextSibling, newIf.`else`?.parent)
        editor?.caretModel?.currentCaret?.let { caret ->
            if (withBraces) {
                caret.moveToOffset(element.endOffset - 1)
                val documentManager = PsiDocumentManager.getInstance(project)
                documentManager.getDocument(element.containingFile)?.let { doc ->
                    documentManager.doPostponedOperationsAndUnblockDocument(doc)
                    editor.executeEnterHandler()
                }
            } else {
                element.`else`?.textRange?.let {
                    caret.moveToOffset(it.startOffset)
                    caret.setSelection(it.startOffset, it.endOffset)
                }
            }
        }
    }

    companion object : QuickFixesPsiBasedFactory<PsiElement>(PsiElement::class, PsiElementSuitabilityCheckers.ALWAYS_SUITABLE) {
        override fun doCreateQuickFix(psiElement: PsiElement): List<IntentionAction> {
            return listOfNotNull(psiElement.parent?.safeAs<KtIfExpression>()?.let(::AddIfElseBranchFix))
        }
    }
}