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

org.jetbrains.kotlin.js.resolve.diagnostics.JsModuleCallChecker.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-Beta1
Show newest version
/*
 * 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.
 */

package org.jetbrains.kotlin.js.resolve.diagnostics

import com.intellij.psi.PsiElement
import org.jetbrains.kotlin.descriptors.ClassDescriptor
import org.jetbrains.kotlin.descriptors.DeclarationDescriptor
import org.jetbrains.kotlin.resolve.DescriptorUtils
import org.jetbrains.kotlin.resolve.calls.checkers.CallChecker
import org.jetbrains.kotlin.resolve.calls.checkers.CallCheckerContext
import org.jetbrains.kotlin.resolve.calls.model.ResolvedCall
import org.jetbrains.kotlin.resolve.scopes.receivers.ClassValueReceiver

object JsModuleCallChecker : CallChecker {
    override fun check(resolvedCall: ResolvedCall<*>, reportOn: PsiElement, context: CallCheckerContext) {
        checkReifieidTypeParameters(resolvedCall, reportOn, context)

        val callee = extractModuleCallee(resolvedCall) ?: return
        val bindingContext = context.trace.bindingContext
        val containingDescriptor = context.scope.ownerDescriptor
        checkJsModuleUsage(bindingContext, context.trace, containingDescriptor, callee, reportOn)
    }

    private fun extractModuleCallee(call: ResolvedCall<*>): DeclarationDescriptor? {
        val callee = call.resultingDescriptor
        if (DescriptorUtils.isTopLevelDeclaration(callee)) return callee

        val receiver = call.dispatchReceiver ?: return callee
        if (receiver is ClassValueReceiver) return receiver.classQualifier.descriptor

        return null
    }

    private fun checkReifieidTypeParameters(call: ResolvedCall<*>, reportOn: PsiElement, context: CallCheckerContext) {
        val containingDescriptor = context.scope.ownerDescriptor
        val typeParams = call.candidateDescriptor.typeParameters.map { it.original }.withIndex().filter { (_, param) -> param.isReified }
        val typeArguments = call.call.typeArgumentList
                ?.let { args ->
                    typeParams.associate { (index, param) -> param.original to args.arguments.getOrNull(index)?.typeReference }
                }
                .orEmpty()
        for (typeParam in typeParams.map { (_, param) -> param.original }) {
            val argPsi = typeArguments[typeParam] ?: reportOn
            val typeArgument = call.typeArguments[typeParam] ?: continue
            val typeArgumentClass = typeArgument.constructor.declarationDescriptor as? ClassDescriptor ?: continue
            checkJsModuleUsage(context.trace.bindingContext, context.trace, containingDescriptor, typeArgumentClass, argPsi)
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy