kotlin.reflect.full.KClassifiers.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-reflect Show documentation
Show all versions of kotlin-reflect Show documentation
Kotlin Full Reflection Library
/*
* Copyright 2010-2016 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.
*/
@file:JvmName("KClassifiers")
package kotlin.reflect.full
import org.jetbrains.kotlin.descriptors.annotations.Annotations
import org.jetbrains.kotlin.types.*
import kotlin.reflect.KClassifier
import kotlin.reflect.KType
import kotlin.reflect.KTypeProjection
import kotlin.reflect.KVariance
import kotlin.reflect.jvm.internal.KClassifierImpl
import kotlin.reflect.jvm.internal.KTypeImpl
import kotlin.reflect.jvm.internal.KotlinReflectionInternalError
/**
* Creates a [KType] instance with the given classifier, type arguments, nullability and annotations.
* If the number of passed type arguments is not equal to the total number of type parameters of a classifier,
* an exception is thrown. If any of the arguments does not satisfy the bounds of the corresponding type parameter,
* an exception is thrown.
*
* For classifiers representing type parameters, the type argument list must always be empty.
* For classes, the type argument list should contain arguments for the type parameters of the class. If the class is `inner`,
* the list should follow with arguments for the type parameters of its outer class, and so forth until a class is
* not `inner`, or is declared on the top level.
*/
@SinceKotlin("1.1")
fun KClassifier.createType(
arguments: List = emptyList(),
nullable: Boolean = false,
annotations: List = emptyList()
): KType {
val descriptor = (this as? KClassifierImpl)?.descriptor
?: throw KotlinReflectionInternalError("Cannot create type for an unsupported classifier: $this (${this.javaClass})")
val typeConstructor = descriptor.typeConstructor
val parameters = typeConstructor.parameters
if (parameters.size != arguments.size) {
throw IllegalArgumentException("Class declares ${parameters.size} type parameters, but ${arguments.size} were provided.")
}
// TODO: throw exception if argument does not satisfy bounds
val typeAnnotations =
if (annotations.isEmpty()) Annotations.EMPTY
else Annotations.EMPTY // TODO: support type annotations
return KTypeImpl(createKotlinType(typeAnnotations, typeConstructor, arguments, nullable))
}
private fun createKotlinType(
typeAnnotations: Annotations, typeConstructor: TypeConstructor, arguments: List, nullable: Boolean
): SimpleType {
val parameters = typeConstructor.parameters
return KotlinTypeFactory.simpleType(typeAnnotations, typeConstructor, arguments.mapIndexed { index, typeProjection ->
val type = (typeProjection.type as KTypeImpl?)?.type
when (typeProjection.variance) {
KVariance.INVARIANT -> TypeProjectionImpl(Variance.INVARIANT, type!!)
KVariance.IN -> TypeProjectionImpl(Variance.IN_VARIANCE, type!!)
KVariance.OUT -> TypeProjectionImpl(Variance.OUT_VARIANCE, type!!)
null -> StarProjectionImpl(parameters[index])
}
}, nullable)
}
/**
* Creates an instance of [KType] with the given classifier, substituting all its type parameters with star projections.
* The resulting type is not marked as nullable and does not have any annotations.
*
* @see [KClassifier.createType]
*/
@SinceKotlin("1.1")
val KClassifier.starProjectedType: KType
get() {
val descriptor = (this as? KClassifierImpl)?.descriptor
?: return createType()
val typeParameters = descriptor.typeConstructor.parameters
if (typeParameters.isEmpty()) return createType() // TODO: optimize, get defaultType from ClassDescriptor
return createType(typeParameters.map { KTypeProjection.STAR })
}