org.jetbrains.kotlin.resolve.AllUnderImportScope.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-compiler-embeddable Show documentation
Show all versions of kotlin-compiler-embeddable Show documentation
the Kotlin compiler embeddable
/*
* Copyright 2010-2015 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jetbrains.kotlin.resolve
import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.incremental.components.LookupLocation
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.resolve.scopes.*
import org.jetbrains.kotlin.util.collectionUtils.flatMapScopes
import org.jetbrains.kotlin.util.collectionUtils.forEachScope
import org.jetbrains.kotlin.utils.Printer
import org.jetbrains.kotlin.utils.SmartList
class AllUnderImportScope private constructor(
descriptor: DeclarationDescriptor,
excludedImportNames: Collection,
private val scope1: MemberScope,
private val scope2: MemberScope?
) : BaseImportingScope(null) {
private val excludedNames: Set = if (excludedImportNames.isEmpty()) { // optimization
emptySet()
} else {
val fqName = DescriptorUtils.getFqNameSafe(descriptor)
// toSet() is used here instead mapNotNullTo(hashSetOf()) because it results in not keeping empty sets as separate instances
excludedImportNames.mapNotNull { if (it.parent() == fqName) it.shortName() else null }.toSet()
}
override fun computeImportedNames(): Set? {
val names1 = scope1.computeAllNames()
return when {
scope2 == null -> names1
names1 == null -> null
else -> {
val names2 = scope2.computeAllNames()
when {
names2 == null -> null
names1.isEmpty() -> names2
else -> names1.toMutableSet().also {
it.addAll(names2)
}
}
}
}
}
override fun getContributedDescriptors(
kindFilter: DescriptorKindFilter,
nameFilter: (Name) -> Boolean,
changeNamesForAliased: Boolean
): Collection {
val nameFilterToUse = if (excludedNames.isEmpty()) { // optimization
nameFilter
} else {
{ it !in excludedNames && nameFilter(it) }
}
val noPackagesKindFilter = kindFilter.withoutKinds(DescriptorKindFilter.PACKAGES_MASK)
val result = SmartList()
forEachScope(scope1, scope2) { scope ->
scope.getContributedDescriptors(noPackagesKindFilter, nameFilterToUse)
.filterTo(result) { it !is PackageViewDescriptor }
}
return result
}
override fun getContributedClassifier(name: Name, location: LookupLocation): ClassifierDescriptor? {
if (name in excludedNames) return null
val classifier1 = scope1.getContributedClassifier(name, location)
val classifier2 = scope2?.getContributedClassifier(name, location)
return if (classifier1 != null && classifier2 != null) null else (classifier1 ?: classifier2)
}
override fun getContributedVariables(name: Name, location: LookupLocation): Collection {
if (name in excludedNames) return emptyList()
return flatMapScopes(scope1, scope2) { it.getContributedVariables(name, location) }
}
override fun getContributedFunctions(name: Name, location: LookupLocation): Collection {
if (name in excludedNames) return emptyList()
return flatMapScopes(scope1, scope2) { it.getContributedFunctions(name, location) }
}
override fun recordLookup(name: Name, location: LookupLocation) {
scope1.recordLookup(name, location)
scope2?.recordLookup(name, location)
}
override fun printStructure(p: Printer) {
p.println(this::class.java.simpleName)
}
companion object {
fun create(descriptor: DeclarationDescriptor, excludedImportNames: Collection): ImportingScope {
val scope1 =
if (descriptor is ClassDescriptor) {
descriptor.staticScope
} else {
assert(descriptor is PackageViewDescriptor) {
"Must be class or package view descriptor: $descriptor"
}
(descriptor as PackageViewDescriptor).memberScope
}
val scope2 =
if (descriptor is ClassDescriptor) {
descriptor.unsubstitutedInnerClassesScope.takeIf { it !== MemberScope.Empty }
} else null
return if (scope1 === MemberScope.Empty) {
if (scope2 == null || scope2 === MemberScope.Empty) ImportingScope.Empty
else AllUnderImportScope(descriptor, excludedImportNames, scope2, null)
} else AllUnderImportScope(descriptor, excludedImportNames, scope1, scope2)
}
}
}