All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jetbrains.kotlin.load.java.lazy.descriptors.SyntheticJavaClassDescriptor.kt Maven / Gradle / Ivy

There is a newer version: 2.0.20-RC
Show newest version
/*
 * Copyright 2000-2018 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.load.java.lazy.descriptors

import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.descriptors.annotations.Annotations
import org.jetbrains.kotlin.descriptors.impl.AbstractClassDescriptor
import org.jetbrains.kotlin.load.java.descriptors.JavaClassDescriptor
import org.jetbrains.kotlin.load.java.lazy.LazyJavaResolverContext
import org.jetbrains.kotlin.load.java.lazy.childForClassOrPackage
import org.jetbrains.kotlin.load.java.structure.*
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.resolve.descriptorUtil.fqNameSafe
import org.jetbrains.kotlin.resolve.descriptorUtil.fqNameUnsafe
import org.jetbrains.kotlin.resolve.scopes.InnerClassesScopeWrapper
import org.jetbrains.kotlin.resolve.scopes.MemberScope
import org.jetbrains.kotlin.types.ClassTypeConstructorImpl
import org.jetbrains.kotlin.types.KotlinType
import org.jetbrains.kotlin.types.SimpleType
import org.jetbrains.kotlin.types.TypeConstructor
import org.jetbrains.kotlin.types.checker.KotlinTypeRefiner


class SyntheticJavaClassDescriptor(
    private val outerContext: LazyJavaResolverContext,
    name: Name,
    private val outerClass: ClassDescriptor,
    private val classKind: ClassKind,
    private val modality: Modality,
    private val visibility: DescriptorVisibility,
    private val isInner: Boolean,
    private val isRecord: Boolean,
    override val annotations: Annotations,
    private val declaredTypeParameters: List,
    private val sealedSubclasses: Collection,
    supertypes: List,
    val attributes: Map
) : AbstractClassDescriptor(
    outerContext.storageManager,
    name
), JavaClassDescriptor {

    companion object {
        @JvmStatic
        private val PUBLIC_METHOD_NAMES_IN_OBJECT = setOf("equals", "hashCode", "getClass", "wait", "notify", "notifyAll", "toString")
    }

    private val jClass = FakeJavaClass()

    private val c: LazyJavaResolverContext = outerContext.childForClassOrPackage(this)

    override fun getKind(): ClassKind = classKind
    override fun getModality(): Modality = modality

    override fun isRecord(): Boolean = isRecord
    override fun getVisibility(): DescriptorVisibility = visibility
    override fun isInner() = isInner
    override fun isData() = false
    override fun isInline() = false
    override fun isCompanionObject() = false
    override fun isExpect() = false
    override fun isActual() = false
    override fun isFun() = false
    override fun isValue() = false

    private val typeConstructor = ClassTypeConstructorImpl(this, declaredTypeParameters, supertypes, c.storageManager)
    override fun getTypeConstructor(): TypeConstructor = typeConstructor

    private val unsubstitutedMemberScope =
        LazyJavaClassMemberScope(c, this, jClass, skipRefinement = true)

    private val scopeHolder =
        ScopesHolderForClass.create(this, c.storageManager, c.components.kotlinTypeChecker.kotlinTypeRefiner) {
            LazyJavaClassMemberScope(
                c, this, jClass,
                skipRefinement = true,
                mainScope = unsubstitutedMemberScope
            )
        }

    override fun getUnsubstitutedMemberScope(kotlinTypeRefiner: KotlinTypeRefiner) = scopeHolder.getScope(kotlinTypeRefiner)

    private val innerClassesScope = InnerClassesScopeWrapper(unsubstitutedMemberScope)
    override fun getUnsubstitutedInnerClassesScope(): MemberScope = innerClassesScope

    private val staticScope = LazyJavaStaticClassScope(c, jClass, this)
    override fun getStaticScope(): MemberScope = staticScope

    override fun getUnsubstitutedPrimaryConstructor(): ClassConstructorDescriptor? = null

    override fun getCompanionObjectDescriptor(): ClassDescriptor? = null

    override fun getUnsubstitutedMemberScope() = super.getUnsubstitutedMemberScope() as LazyJavaClassMemberScope
    override fun getConstructors() = unsubstitutedMemberScope.constructors()

    override fun getDeclaredTypeParameters(): List = declaredTypeParameters

    override fun getDefaultFunctionTypeForSamInterface(): SimpleType? =
        c.components.samConversionResolver.resolveFunctionTypeIfSamInterface(this)

    override fun isDefinitelyNotSamInterface(): Boolean {
        if (classKind != ClassKind.INTERFACE) return true

        // From the definition of function interfaces in the Java specification (pt. 9.8):
        // "methods that are members of I that do not have the same signature as any public instance method of the class Object"
        // It means that if an interface declares `int hashCode()` then the method won't be taken into account when
        // checking if the interface is SAM.
        // We make here a conservative check just filtering out methods by name.
        // If we ignore a method with wrong signature (different from one in Object) it's not very bad,
        // we'll just say that the interface MAY BE a SAM when it's not and then more detailed check will be applied.
        var foundSamMethod = false
        for (method in jClass.methods) {
            if (method.isAbstract && method.typeParameters.isEmpty() &&
                method.name.identifier !in PUBLIC_METHOD_NAMES_IN_OBJECT
            ) {
                // found 2nd method candidate
                if (foundSamMethod) {
                    return true
                }
                foundSamMethod = true
            }
        }

        // If we have default methods the interface could be a SAM even while a super interface has more than one abstract method
        if (jClass.methods.any { !it.isAbstract && it.typeParameters.isEmpty() }) return false

        // Check if any of the super-interfaces contain too many methods to be a SAM
        return typeConstructor.supertypes.any {
            (it.constructor.declarationDescriptor as? SyntheticJavaClassDescriptor)?.isDefinitelyNotSamInterface == true
        }
    }

    override fun getSealedSubclasses(): Collection = sealedSubclasses

    override fun getValueClassRepresentation(): ValueClassRepresentation? = null

    override fun getContainingDeclaration(): DeclarationDescriptor {
        return outerClass
    }

    private val sourceElement = c.components.sourceElementFactory.source(jClass)

    override fun getSource(): SourceElement = sourceElement

    override fun isExternal(): Boolean = false

    override fun toString() = "Lazy Java class ${this.fqNameUnsafe}"

    private inner class FakeJavaClass : JavaClass {
        override val name: Name
            get() = [email protected]
        override val isFromSource: Boolean
            get() = false
        override val annotations: Collection
            get() = emptyList()
        override val isDeprecatedInJavaDoc: Boolean
            get() = false

        override fun findAnnotation(fqName: FqName): JavaAnnotation? = null

        override val isAbstract: Boolean
            get() = modality == Modality.ABSTRACT
        override val isStatic: Boolean
            get() = !isInner
        override val isFinal: Boolean
            get() = modality == Modality.FINAL
        override val visibility: Visibility
            get() = [email protected]
        override val typeParameters: List
            get() = emptyList()
        override val fqName: FqName
            get() = [email protected]
        override val supertypes: Collection
            get() = emptyList()
        override val innerClassNames: Collection
            get() = emptyList()

        override fun findInnerClass(name: Name): JavaClass? = null

        override val outerClass: JavaClass?
            get() = null
        override val isInterface: Boolean
            get() = classKind == ClassKind.INTERFACE
        override val isAnnotationType: Boolean
            get() = classKind == ClassKind.ANNOTATION_CLASS
        override val isEnum: Boolean
            get() = classKind == ClassKind.ENUM_CLASS
        override val isRecord: Boolean
            get() = [email protected]
        override val isSealed: Boolean
            get() = modality == Modality.SEALED
        override val permittedTypes: Collection
            get() = emptyList()
        override val lightClassOriginKind: LightClassOriginKind?
            get() = null
        override val methods: Collection
            get() = emptyList()
        override val fields: Collection
            get() = emptyList()
        override val constructors: Collection
            get() = emptyList()
        override val recordComponents: Collection
            get() = emptyList()

        override fun hasDefaultConstructor(): Boolean = false
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy