org.jetbrains.kotlin.fir.scopes.ImportingScopes.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package org.jetbrains.kotlin.fir.scopes
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.declarations.FirFile
import org.jetbrains.kotlin.fir.declarations.FirResolvePhase
import org.jetbrains.kotlin.fir.importTracker
import org.jetbrains.kotlin.fir.packageFqName
import org.jetbrains.kotlin.fir.reportImportDirectives
import org.jetbrains.kotlin.fir.resolve.ScopeSession
import org.jetbrains.kotlin.fir.resolve.scopeSessionKey
import org.jetbrains.kotlin.fir.scopes.impl.*
import org.jetbrains.kotlin.fir.symbols.lazyResolveToPhase
import org.jetbrains.kotlin.name.FqName
private val ALL_IMPORTS = scopeSessionKey()
private val DEFAULT_STAR_IMPORT = scopeSessionKey()
private val DEFAULT_SIMPLE_IMPORT = scopeSessionKey()
private data class DefaultStarImportKey(val priority: DefaultImportPriority, val excludedImportNames: Set)
fun createImportingScopes(
file: FirFile,
session: FirSession,
scopeSession: ScopeSession,
useCaching: Boolean = true,
): List = if (useCaching) {
scopeSession.getOrBuild(file, ALL_IMPORTS) {
ListStorageFirScope(computeImportingScopes(file, session, scopeSession))
}.result
} else {
computeImportingScopes(file, session, scopeSession)
}
internal fun computeImportingScopes(
file: FirFile,
session: FirSession,
scopeSession: ScopeSession,
includeDefaultImports: Boolean = true,
includePackageImport: Boolean = true
): List {
file.lazyResolveToPhase(FirResolvePhase.IMPORTS)
val excludedImportNames =
file.imports.filter { it.aliasName != null }.mapNotNullTo(hashSetOf()) { it.importedFqName }.ifEmpty { emptySet() }
val excludedNamesInPackage =
excludedImportNames.mapNotNullTo(mutableSetOf()) {
if (it.parent() == file.packageFqName) it.shortName() else null
}
session.importTracker?.let { tracker ->
file.imports.map { import ->
tracker.reportImportDirectives(file.sourceFile?.path, import.importedFqName?.asString())
}
}
return buildList {
if (includeDefaultImports) {
this += FirDefaultStarImportingScope(
scopeSession.getOrBuild(DefaultStarImportKey(DefaultImportPriority.HIGH, excludedImportNames), DEFAULT_STAR_IMPORT) {
FirSingleLevelDefaultStarImportingScope(session, scopeSession, DefaultImportPriority.HIGH, excludedImportNames)
},
scopeSession.getOrBuild(DefaultStarImportKey(DefaultImportPriority.LOW, excludedImportNames), DEFAULT_STAR_IMPORT) {
FirSingleLevelDefaultStarImportingScope(session, scopeSession, DefaultImportPriority.LOW, excludedImportNames)
},
)
this += scopeSession.getOrBuild(DefaultImportPriority.KOTLIN_THROWS, DEFAULT_SIMPLE_IMPORT) {
FirDefaultSimpleImportingScope(session, scopeSession, priority = DefaultImportPriority.KOTLIN_THROWS)
}
}
this += FirExplicitStarImportingScope(file.imports, session, scopeSession, excludedImportNames)
if (includeDefaultImports) {
this += scopeSession.getOrBuild(DefaultImportPriority.LOW, DEFAULT_SIMPLE_IMPORT) {
FirDefaultSimpleImportingScope(session, scopeSession, priority = DefaultImportPriority.LOW)
}
this += scopeSession.getOrBuild(DefaultImportPriority.HIGH, DEFAULT_SIMPLE_IMPORT) {
FirDefaultSimpleImportingScope(session, scopeSession, priority = DefaultImportPriority.HIGH)
}
}
if (includePackageImport) {
this += when {
excludedNamesInPackage.isEmpty() ->
// Supposed to be the most common branch, so we cache it
scopeSession.getOrBuild(file.packageFqName, PACKAGE_MEMBER) {
FirPackageMemberScope(file.packageFqName, session, excludedNames = emptySet())
}
else ->
FirPackageMemberScope(file.packageFqName, session, excludedNames = excludedNamesInPackage)
}
}
// TODO: explicit simple importing scope should have highest priority (higher than inner scopes added in process)
this += FirExplicitSimpleImportingScope(file.imports, session, scopeSession)
}
}
private class ListStorageFirScope(val result: List) : FirScope()
© 2015 - 2024 Weber Informatics LLC | Privacy Policy