org.jetbrains.kotlin.fir.EnumClassUtils.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2020 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
import org.jetbrains.kotlin.KtFakeSourceElementKind
import org.jetbrains.kotlin.builtins.StandardNames.DEFAULT_VALUE_PARAMETER
import org.jetbrains.kotlin.builtins.StandardNames.ENUM_ENTRIES
import org.jetbrains.kotlin.builtins.StandardNames.ENUM_VALUES
import org.jetbrains.kotlin.builtins.StandardNames.ENUM_VALUE_OF
import org.jetbrains.kotlin.descriptors.Modality
import org.jetbrains.kotlin.descriptors.Visibilities
import org.jetbrains.kotlin.fakeElement
import org.jetbrains.kotlin.fir.declarations.FirDeclarationOrigin
import org.jetbrains.kotlin.fir.declarations.FirDeclarationStatus
import org.jetbrains.kotlin.fir.declarations.builder.FirRegularClassBuilder
import org.jetbrains.kotlin.fir.declarations.builder.buildProperty
import org.jetbrains.kotlin.fir.declarations.builder.buildSimpleFunction
import org.jetbrains.kotlin.fir.declarations.builder.buildValueParameter
import org.jetbrains.kotlin.fir.declarations.impl.FirDeclarationStatusImpl
import org.jetbrains.kotlin.fir.declarations.impl.FirDefaultPropertyGetter
import org.jetbrains.kotlin.fir.declarations.impl.FirResolvedDeclarationStatusImpl
import org.jetbrains.kotlin.fir.expressions.builder.buildEmptyExpressionBlock
import org.jetbrains.kotlin.fir.symbols.impl.FirNamedFunctionSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirPropertySymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirValueParameterSymbol
import org.jetbrains.kotlin.fir.types.ConeTypeProjection
import org.jetbrains.kotlin.fir.types.builder.buildResolvedTypeRef
import org.jetbrains.kotlin.fir.types.impl.ConeClassLikeTypeImpl
import org.jetbrains.kotlin.fir.types.toLookupTag
import org.jetbrains.kotlin.name.CallableId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.StandardClassIds
fun FirRegularClassBuilder.generateValuesFunction(
moduleData: FirModuleData,
packageFqName: FqName,
classFqName: FqName,
makeExpect: Boolean = false,
origin: FirDeclarationOrigin = FirDeclarationOrigin.Source
) {
val sourceElement = source?.fakeElement(KtFakeSourceElementKind.EnumGeneratedDeclaration)
declarations += buildSimpleFunction {
source = sourceElement
this.origin = origin
this.moduleData = moduleData
val returnTypeRef = buildResolvedTypeRef {
source = sourceElement
type = ConeClassLikeTypeImpl(
StandardClassIds.Array.toLookupTag(),
arrayOf(
ConeClassLikeTypeImpl(
[email protected](),
ConeTypeProjection.EMPTY_ARRAY,
isNullable = false
)
),
isNullable = false
)
}
this.returnTypeRef = returnTypeRef
name = ENUM_VALUES
this.status = createStatus([email protected]).apply {
isStatic = true
isExpect = makeExpect
}
symbol = FirNamedFunctionSymbol(CallableId(packageFqName, classFqName, ENUM_VALUES))
resolvePhase = [email protected]
body = buildEmptyExpressionBlock().also {
it.replaceConeTypeOrNull(returnTypeRef.type)
}
}.apply {
containingClassForStaticMemberAttr = [email protected]()
}
}
fun FirRegularClassBuilder.generateValueOfFunction(
moduleData: FirModuleData,
packageFqName: FqName,
classFqName: FqName,
makeExpect: Boolean = false,
origin: FirDeclarationOrigin = FirDeclarationOrigin.Source
) {
val sourceElement = source?.fakeElement(KtFakeSourceElementKind.EnumGeneratedDeclaration)
declarations += buildSimpleFunction {
source = sourceElement
this.origin = origin
this.moduleData = moduleData
val returnTypeRef = buildResolvedTypeRef {
source = sourceElement
type = ConeClassLikeTypeImpl(
[email protected](),
emptyArray(),
isNullable = false
)
}
this.returnTypeRef = returnTypeRef
name = ENUM_VALUE_OF
status = createStatus([email protected]).apply {
isStatic = true
isExpect = makeExpect
}
symbol = FirNamedFunctionSymbol(CallableId(packageFqName, classFqName, ENUM_VALUE_OF))
valueParameters += buildValueParameter vp@{
source = sourceElement
containingFunctionSymbol = [email protected]
this.origin = origin
this.moduleData = moduleData
this.returnTypeRef = buildResolvedTypeRef {
source = sourceElement
type = ConeClassLikeTypeImpl(
StandardClassIds.String.toLookupTag(),
emptyArray(),
isNullable = false
)
}
name = DEFAULT_VALUE_PARAMETER
[email protected] = FirValueParameterSymbol(DEFAULT_VALUE_PARAMETER)
isCrossinline = false
isNoinline = false
isVararg = false
resolvePhase = [email protected]
}
resolvePhase = [email protected]
body = buildEmptyExpressionBlock().also {
it.replaceConeTypeOrNull(returnTypeRef.type)
}
}.apply {
containingClassForStaticMemberAttr = [email protected]()
}
}
fun FirRegularClassBuilder.generateEntriesGetter(
moduleData: FirModuleData,
packageFqName: FqName,
classFqName: FqName,
makeExpect: Boolean = false,
origin: FirDeclarationOrigin = FirDeclarationOrigin.Source
) {
val sourceElement = source?.fakeElement(KtFakeSourceElementKind.EnumGeneratedDeclaration)
declarations += buildProperty {
source = sourceElement
isVar = false
isLocal = false
this.origin = origin
this.moduleData = moduleData
returnTypeRef = buildResolvedTypeRef {
source = sourceElement
type = ConeClassLikeTypeImpl(
StandardClassIds.EnumEntries.toLookupTag(),
arrayOf(
ConeClassLikeTypeImpl([email protected](), ConeTypeProjection.EMPTY_ARRAY, isNullable = false)
),
isNullable = false
)
}
name = ENUM_ENTRIES
this.status = createStatus([email protected]).apply {
isStatic = true
isExpect = makeExpect
}
symbol = FirPropertySymbol(CallableId(packageFqName, classFqName, ENUM_ENTRIES))
resolvePhase = [email protected]
getter = FirDefaultPropertyGetter(
sourceElement?.fakeElement(KtFakeSourceElementKind.EnumGeneratedDeclaration),
moduleData, origin, returnTypeRef.copyWithNewSourceKind(KtFakeSourceElementKind.EnumGeneratedDeclaration),
Visibilities.Public, symbol, resolvePhase = [email protected]
).apply {
this.status = createStatus([email protected]).apply {
isStatic = true
}
}
}.apply {
containingClassForStaticMemberAttr = [email protected]()
}
}
private fun createStatus(parentStatus: FirDeclarationStatus): FirDeclarationStatusImpl {
val parentEffectiveVisibility = (parentStatus as? FirResolvedDeclarationStatusImpl)?.effectiveVisibility
return if (parentEffectiveVisibility != null) {
FirResolvedDeclarationStatusImpl(Visibilities.Public, Modality.FINAL, parentEffectiveVisibility)
} else {
FirDeclarationStatusImpl(Visibilities.Public, Modality.FINAL)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy