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

main.seskar.compiler.key.backend.DefaultKeyTransformer.kt Maven / Gradle / Ivy

package seskar.compiler.key.backend

import org.jetbrains.kotlin.backend.common.extensions.IrPluginContext
import org.jetbrains.kotlin.ir.IrElement
import org.jetbrains.kotlin.ir.IrStatement
import org.jetbrains.kotlin.ir.declarations.IrFile
import org.jetbrains.kotlin.ir.expressions.IrCall
import org.jetbrains.kotlin.ir.expressions.impl.IrCallImpl
import org.jetbrains.kotlin.ir.expressions.impl.IrCompositeImpl
import org.jetbrains.kotlin.ir.expressions.impl.IrConstImpl
import org.jetbrains.kotlin.ir.types.defaultType
import org.jetbrains.kotlin.ir.util.hasAnnotation
import org.jetbrains.kotlin.ir.visitors.IrElementTransformer
import org.jetbrains.kotlin.name.CallableId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name

private val ELEMENT_BUILDER = FqName("react.ElementBuilder")

private val SET_DEFAULT_KEY = CallableId(
    packageName = FqName("react"),
    className = null,
    callableName = Name.identifier("setDefaultKey"),
)

internal class DefaultKeyTransformer(
    private val context: IrPluginContext,
) : IrElementTransformer {
    override fun visitFile(
        declaration: IrFile,
        data: KeyProvider?,
    ): IrFile =
        super.visitFile(
            declaration = declaration,
            data = KeyProvider(declaration.fileEntry),
        )

    override fun visitCall(
        expression: IrCall,
        data: KeyProvider?,
    ): IrElement {
        val originalCall = super.visitCall(expression, data)
        data ?: return originalCall

        val keyCall = keyCall(expression, data)
            ?: return originalCall

        // TODO: check how to avoid
        originalCall as IrStatement

        return IrCompositeImpl(
            startOffset = expression.startOffset,
            endOffset = expression.endOffset,
            type = expression.type,
            origin = expression.origin,
            statements = listOf(
                keyCall,
                originalCall,
            )
        )
    }

    private fun keyCall(
        expression: IrCall,
        keyProvider: KeyProvider,
    ): IrCall? {
        val dispatchReceiver = expression.dispatchReceiver
            ?: return null

        if (!expression.symbol.owner.hasAnnotation(ELEMENT_BUILDER))
            return null

        val setDefaultKey = context.referenceFunctions(SET_DEFAULT_KEY).single()

        val key = keyProvider.get(expression)

        val call = IrCallImpl.fromSymbolOwner(
            startOffset = expression.startOffset,
            endOffset = expression.endOffset,
            symbol = setDefaultKey,
        )

        val defaultKey = IrConstImpl.string(
            startOffset = expression.startOffset,
            endOffset = expression.endOffset,
            type = context.symbols.string.defaultType,
            value = "@rdk/$key",
        )

        call.putValueArgument(0, dispatchReceiver)
        call.putValueArgument(1, defaultKey)

        return call
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy