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

org.jetbrains.kotlin.resolve.jvm.ReplaceWithSupertypeAnonymousTypeTransformer.kt Maven / Gradle / Ivy

There is a newer version: 2.0.20-RC
Show newest version
/*
 * Copyright 2010-2023 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.resolve.jvm

import org.jetbrains.kotlin.builtins.KotlinBuiltIns
import org.jetbrains.kotlin.descriptors.ClassDescriptor
import org.jetbrains.kotlin.descriptors.DeclarationDescriptorWithVisibility
import org.jetbrains.kotlin.resolve.DeclarationSignatureAnonymousTypeTransformer
import org.jetbrains.kotlin.resolve.DescriptorUtils
import org.jetbrains.kotlin.types.KotlinType
import org.jetbrains.kotlin.types.TypeProjectionImpl
import org.jetbrains.kotlin.types.replace
import org.jetbrains.kotlin.types.typeUtil.builtIns

open class ReplaceWithSupertypeAnonymousTypeTransformer : DeclarationSignatureAnonymousTypeTransformer {
    override fun transformAnonymousType(descriptor: DeclarationDescriptorWithVisibility, type: KotlinType): KotlinType? =
        if (!DescriptorUtils.isLocal(descriptor))
            replaceAnonymousTypeWithSuperType(type)
        else type
}

fun replaceAnonymousTypeWithSuperType(type: KotlinType): KotlinType {
    val declaration = type.constructor.declarationDescriptor as? ClassDescriptor ?: return type

    if (KotlinBuiltIns.isArray(type)) {
        val elementTypeProjection = type.arguments.singleOrNull()
        if (elementTypeProjection != null && !elementTypeProjection.isStarProjection) {
            return type.builtIns.getArrayType(
                elementTypeProjection.projectionKind,
                replaceAnonymousTypeWithSuperType(elementTypeProjection.type)
            )
        }
    }

    val actualType = when {
        DescriptorUtils.isAnonymousObject(declaration) || DescriptorUtils.isLocal(declaration) -> {
            if (type.constructor.supertypes.size == 1) {
                replaceAnonymousTypeWithSuperType(type.constructor.supertypes.iterator().next())
            } else {
                /*
                    Frontend reports an error on public properties in this case,
                    but we ignore errors when making stubs, so there should be a reasonable fallback.
                 */
                type.builtIns.anyType
            }
        }

        else -> type
    }

    if (actualType.arguments.isEmpty()) return actualType

    val arguments = actualType.arguments.map { typeArg ->
        if (typeArg.isStarProjection)
            return@map typeArg

        TypeProjectionImpl(typeArg.projectionKind, replaceAnonymousTypeWithSuperType(typeArg.type))
    }

    return actualType.replace(newArguments = arguments)
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy