summaryrefslogtreecommitdiff
path: root/plugins/kotlin/idea/src/org/jetbrains/kotlin/idea/KotlinPairMatcher.kt
blob: c6975b87215b86462e9f3a16ed9f44f43e1732cc (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
// Copyright 2000-2021 JetBrains s.r.o. and contributors. Use of this source code is governed by the Apache 2.0 license that can be found in the LICENSE file.

package org.jetbrains.kotlin.idea

import com.intellij.codeInsight.hint.DeclarationRangeUtil
import com.intellij.lang.BracePair
import com.intellij.lang.PairedBraceMatcher
import com.intellij.psi.PsiFile
import com.intellij.psi.tree.IElementType
import org.jetbrains.kotlin.lexer.KtTokens
import org.jetbrains.kotlin.psi.KtBlockExpression
import org.jetbrains.kotlin.psi.KtClassBody

class KotlinPairMatcher : PairedBraceMatcher {
    private val pairs = arrayOf(
        BracePair(KtTokens.LPAR, KtTokens.RPAR, false),
        BracePair(KtTokens.LONG_TEMPLATE_ENTRY_START, KtTokens.LONG_TEMPLATE_ENTRY_END, false),
        BracePair(KtTokens.LBRACE, KtTokens.RBRACE, true),
        BracePair(KtTokens.LBRACKET, KtTokens.RBRACKET, false),
        BracePair(KtTokens.LT, KtTokens.GT, false),
    )

    override fun getPairs(): Array<BracePair> = pairs

    override fun isPairedBracesAllowedBeforeType(lbraceType: IElementType, contextType: IElementType?): Boolean {
        return if (lbraceType == KtTokens.LONG_TEMPLATE_ENTRY_START) {
            // KotlinTypedHandler insert paired brace in this case
            false
        } else KtTokens.WHITE_SPACE_OR_COMMENT_BIT_SET.contains(contextType)
                || contextType === KtTokens.COLON
                || contextType === KtTokens.SEMICOLON
                || contextType === KtTokens.COMMA
                || contextType === KtTokens.RPAR
                || contextType === KtTokens.RBRACKET
                || contextType === KtTokens.RBRACE
                || contextType === KtTokens.LBRACE
                || contextType === KtTokens.LONG_TEMPLATE_ENTRY_END

    }

    override fun getCodeConstructStart(file: PsiFile, openingBraceOffset: Int): Int {
        val element = file.findElementAt(openingBraceOffset)
        if (element == null || element is PsiFile) return openingBraceOffset
        return when (val parent = element.parent) {
            is KtClassBody, is KtBlockExpression ->
                DeclarationRangeUtil.getPossibleDeclarationAtRange(parent.parent)?.startOffset ?: openingBraceOffset

            else -> openingBraceOffset
        }
    }
}